[COMMON] sched: fair/ehmp: Consider cpu capacity when idlest group finds.
authorlakkyung.jung <lakkyung.jung@samsung.com>
Tue, 13 Mar 2018 11:35:22 +0000 (20:35 +0900)
committerChungwoo Park <cww.park@samsung.com>
Mon, 21 May 2018 08:30:47 +0000 (17:30 +0900)
Change-Id: I2f168aced04ade3321adcd3408ce2da344d4ab3b
Signed-off-by: lakkyung.jung <lakkyung.jung@samsung.com>
include/linux/ehmp.h
kernel/sched/ehmp.c
kernel/sched/fair.c

index 3b2f4d8beedeaf6a813906cd30bd69d5ac678e1d..6d52f62efd6fb05e7052f40ffd7cc206a657553f 100644 (file)
@@ -31,6 +31,8 @@ struct gb_qos_request {
 };
 
 #ifdef CONFIG_SCHED_EHMP
+extern struct sched_group *exynos_fit_idlest_group(struct sched_domain *sd,
+               struct task_struct *p);
 extern void exynos_init_entity_util_avg(struct sched_entity *se);
 extern int exynos_need_active_balance(enum cpu_idle_type idle,
                struct sched_domain *sd, int src_cpu, int dst_cpu);
@@ -57,6 +59,8 @@ extern void gb_qos_update_request(struct gb_qos_request *req, u32 new_value);
 
 extern void request_kernel_prefer_perf(int grp_idx, int enable);
 #else
+static inline struct sched_group *exynos_fit_idlest_group(struct sched_domain *sd,
+               struct task_struct *p) { return NULL; }
 static inline void exynos_init_entity_util_avg(struct sched_entity *se) { }
 static inline int exynos_need_active_balance(enum cpu_idle_type idle,
                struct sched_domain *sd, int src_cpu, int dst_cpu) { return 0; }
index b3cd7ecd512d19f30e7940f7c19e5479de278e67..ce396851e9b9cc4c34a5fccb8dfbd8c29971728d 100644 (file)
 #include "sched.h"
 #include "tune.h"
 
+/**********************************************************************
+ * extern functions                                                   *
+ **********************************************************************/
+extern struct sched_entity *__pick_next_entity(struct sched_entity *se);
+extern unsigned long boosted_task_util(struct task_struct *task);
+extern unsigned long capacity_curr_of(int cpu);
+extern int find_best_target(struct task_struct *p, int *backup_cpu,
+                                  bool boosted, bool prefer_idle);
+extern u64 decay_load(u64 val, u64 n);
+extern int start_cpu(bool boosted);
+
 static unsigned long task_util(struct task_struct *p)
 {
        return p->se.avg.util_avg;
@@ -32,6 +43,11 @@ static inline struct sched_entity *se_of(struct sched_avg *sa)
        return container_of(sa, struct sched_entity, avg);
 }
 
+static inline int task_fits(struct task_struct *p, long capacity)
+{
+       return capacity * 1024 > boosted_task_util(p) * 1248;
+}
+
 #define entity_is_cfs_rq(se)   (se->my_q)
 #define entity_is_task(se)     (!se->my_q)
 #define LOAD_AVG_MAX           47742
@@ -178,6 +194,32 @@ bool cpu_overutilized(int cpu);
 #define lb_sd_parent(sd) \
        (sd->parent && sd->parent->groups != sd->parent->groups->next)
 
+struct sched_group *
+exynos_fit_idlest_group(struct sched_domain *sd, struct task_struct *p)
+{
+       struct sched_group *group = sd->groups;
+       struct sched_group *fit_group = NULL;
+       unsigned long fit_capacity = ULONG_MAX;
+
+       do {
+               int i;
+
+               /* Skip over this group if it has no CPUs allowed */
+               if (!cpumask_intersects(sched_group_span(group),
+                                       &p->cpus_allowed))
+                       continue;
+
+               for_each_cpu(i, sched_group_span(group)) {
+                       if (capacity_of(i) < fit_capacity && task_fits(p, capacity_of(i))) {
+                               fit_capacity = capacity_of(i);
+                               fit_group = group;
+                       }
+               }
+       } while (group = group->next, group != sd->groups);
+
+       return fit_group;
+}
+
 static inline int
 check_cpu_capacity(struct rq *rq, struct sched_domain *sd)
 {
@@ -1092,7 +1134,6 @@ ontime_select_target_cpu(struct sched_group *sg, const struct cpumask *mask)
 
 #define TASK_TRACK_COUNT       5
 
-extern struct sched_entity *__pick_next_entity(struct sched_entity *se);
 static struct task_struct *
 ontime_pick_heavy_task(struct sched_entity *se, struct cpumask *dst_cpus,
                                                int *boost_migration)
@@ -1508,8 +1549,6 @@ static void ontime_update_next_balance(int cpu, struct ontime_avg *oa)
 
 #define cap_scale(v, s) ((v)*(s) >> SCHED_CAPACITY_SHIFT)
 
-extern u64 decay_load(u64 val, u64 n);
-
 static u32 __accumulate_pelt_segments(u64 periods, u32 d1, u32 d3)
 {
        u32 c1, c2, c3 = d3;
@@ -1608,12 +1647,6 @@ pure_initcall(init_ontime);
 /**********************************************************************
  * cpu selection                                                      *
  **********************************************************************/
-extern unsigned long boosted_task_util(struct task_struct *task);
-extern unsigned long capacity_curr_of(int cpu);
-extern int find_best_target(struct task_struct *p, int *backup_cpu,
-                                  bool boosted, bool prefer_idle);
-extern int start_cpu(bool boosted);
-
 #define EAS_CPU_PRV    0
 #define EAS_CPU_NXT    1
 #define EAS_CPU_BKP    2
index 6c224355bf92f02ba0e6b4cd173a55d7b15cf2e7..655d5a6a311589b4f4cac20e2007d9cc80121842 100644 (file)
@@ -6277,6 +6277,12 @@ find_idlest_group(struct sched_domain *sd, struct task_struct *p,
        unsigned long imbalance = scale_load_down(NICE_0_LOAD) *
                                (sd->imbalance_pct-100) / 100;
 
+       if (sched_feat(EXYNOS_HMP)) {
+               idlest = exynos_fit_idlest_group(sd, p);
+               if (idlest)
+                       return idlest;
+       }
+
        if (sd_flag & SD_BALANCE_WAKE)
                load_idx = sd->wake_idx;