diff options
Diffstat (limited to 'kernel/sched/sched.h')
| -rw-r--r-- | kernel/sched/sched.h | 848 | 
1 files changed, 832 insertions, 16 deletions
| diff --git a/kernel/sched/sched.h b/kernel/sched/sched.h index 3db67eb07bdd..ca2294d06f44 100644 --- a/kernel/sched/sched.h +++ b/kernel/sched/sched.h @@ -27,14 +27,13 @@ extern unsigned long calc_load_update;  extern atomic_long_t calc_load_tasks;  extern void calc_global_load_tick(struct rq *this_rq); +  extern long calc_load_fold_active(struct rq *this_rq);  #ifdef CONFIG_SMP  extern void update_cpu_load_active(struct rq *this_rq); -extern void check_for_migration(struct rq *rq, struct task_struct *p);  #else  static inline void update_cpu_load_active(struct rq *this_rq) { } -static inline void check_for_migration(struct rq *rq, struct task_struct *p) { }  #endif  /* @@ -242,6 +241,10 @@ struct cfs_bandwidth {  struct task_group {  	struct cgroup_subsys_state css; +#ifdef CONFIG_SCHED_HMP +	bool upmigrate_discouraged; +#endif +  #ifdef CONFIG_FAIR_GROUP_SCHED  	/* schedulable entities of this group on each cpu */  	struct sched_entity **se; @@ -310,7 +313,7 @@ extern int tg_nop(struct task_group *tg, void *data);  extern void free_fair_sched_group(struct task_group *tg);  extern int alloc_fair_sched_group(struct task_group *tg, struct task_group *parent); -extern void unregister_fair_sched_group(struct task_group *tg, int cpu); +extern void unregister_fair_sched_group(struct task_group *tg);  extern void init_tg_cfs_entry(struct task_group *tg, struct cfs_rq *cfs_rq,  			struct sched_entity *se, int cpu,  			struct sched_entity *parent); @@ -347,12 +350,96 @@ static inline void set_task_rq_fair(struct sched_entity *se,  #endif /* CONFIG_SMP */  #endif /* CONFIG_FAIR_GROUP_SCHED */ +extern struct task_group *css_tg(struct cgroup_subsys_state *css);  #else /* CONFIG_CGROUP_SCHED */  struct cfs_bandwidth { };  #endif	/* CONFIG_CGROUP_SCHED */ +#ifdef CONFIG_SCHED_HMP + +#define NUM_TRACKED_WINDOWS 2 +#define NUM_LOAD_INDICES 1000 + +struct hmp_sched_stats { +	int nr_big_tasks; +	u64 cumulative_runnable_avg; +	u64 pred_demands_sum; +}; + +struct load_subtractions { +	u64 window_start; +	u64 subs; +	u64 new_subs; +}; + +struct group_cpu_time { +	u64 curr_runnable_sum; +	u64 prev_runnable_sum; +	u64 nt_curr_runnable_sum; +	u64 nt_prev_runnable_sum; +}; + +struct sched_cluster { +	raw_spinlock_t load_lock; +	struct list_head list; +	struct cpumask cpus; +	int id; +	int max_power_cost; +	int min_power_cost; +	int max_possible_capacity; +	int capacity; +	int efficiency; /* Differentiate cpus with different IPC capability */ +	int load_scale_factor; +	unsigned int exec_scale_factor; +	/* +	 * max_freq = user maximum +	 * max_mitigated_freq = thermal defined maximum +	 * max_possible_freq = maximum supported by hardware +	 */ +	unsigned int cur_freq, max_freq, max_mitigated_freq, min_freq; +	unsigned int max_possible_freq; +	bool freq_init_done; +	int dstate, dstate_wakeup_latency, dstate_wakeup_energy; +	unsigned int static_cluster_pwr_cost; +	int notifier_sent; +	bool wake_up_idle; +	atomic64_t last_cc_update; +	atomic64_t cycles; +}; + +extern unsigned long all_cluster_ids[]; + +static inline int cluster_first_cpu(struct sched_cluster *cluster) +{ +	return cpumask_first(&cluster->cpus); +} + +struct related_thread_group { +	int id; +	raw_spinlock_t lock; +	struct list_head tasks; +	struct list_head list; +	struct sched_cluster *preferred_cluster; +	struct rcu_head rcu; +	u64 last_update; +}; + +extern struct list_head cluster_head; +extern int num_clusters; +extern struct sched_cluster *sched_cluster[NR_CPUS]; + +struct cpu_cycle { +	u64 cycles; +	u64 time; +}; + +#define for_each_sched_cluster(cluster) \ +	list_for_each_entry_rcu(cluster, &cluster_head, list) + +#endif /* CONFIG_SCHED_HMP */ +  /* CFS-related fields in a runqueue */  struct cfs_rq {  	struct load_weight load; @@ -421,11 +508,12 @@ struct cfs_rq {  	struct list_head leaf_cfs_rq_list;  	struct task_group *tg;	/* group that "owns" this runqueue */ -#ifdef CONFIG_SCHED_WALT -	u64 cumulative_runnable_avg; +#ifdef CONFIG_CFS_BANDWIDTH + +#ifdef CONFIG_SCHED_HMP +	struct hmp_sched_stats hmp_stats;  #endif -#ifdef CONFIG_CFS_BANDWIDTH  	int runtime_enabled;  	u64 runtime_expires;  	s64 runtime_remaining; @@ -693,6 +781,37 @@ struct rq {  	u64 max_idle_balance_cost;  #endif +#ifdef CONFIG_SCHED_HMP +	struct sched_cluster *cluster; +	struct cpumask freq_domain_cpumask; +	struct hmp_sched_stats hmp_stats; + +	int cstate, wakeup_latency, wakeup_energy; +	u64 window_start; +	unsigned long hmp_flags; + +	u64 cur_irqload; +	u64 avg_irqload; +	u64 irqload_ts; +	unsigned int static_cpu_pwr_cost; +	struct task_struct *ed_task; +	struct cpu_cycle cc; +	u64 old_busy_time, old_busy_time_group; +	u64 old_estimated_time; +	u64 curr_runnable_sum; +	u64 prev_runnable_sum; +	u64 nt_curr_runnable_sum; +	u64 nt_prev_runnable_sum; +	struct group_cpu_time grp_time; +	struct load_subtractions load_subs[NUM_TRACKED_WINDOWS]; +	DECLARE_BITMAP_ARRAY(top_tasks_bitmap, +			NUM_TRACKED_WINDOWS, NUM_LOAD_INDICES); +	u8 *top_tasks[NUM_TRACKED_WINDOWS]; +	u8 curr_table; +	int prev_top; +	int curr_top; +#endif +  #ifdef CONFIG_SCHED_WALT  	u64 cumulative_runnable_avg;  	u64 window_start; @@ -706,7 +825,6 @@ struct rq {  	u64 cum_window_demand;  #endif /* CONFIG_SCHED_WALT */ -  #ifdef CONFIG_IRQ_TIME_ACCOUNTING  	u64 prev_irq_time;  #endif @@ -981,6 +1099,648 @@ static inline void sched_ttwu_pending(void) { }  #include "stats.h"  #include "auto_group.h" +enum sched_boost_policy { +	SCHED_BOOST_NONE, +	SCHED_BOOST_ON_BIG, +	SCHED_BOOST_ON_ALL, +}; + +#ifdef CONFIG_SCHED_HMP + +#define WINDOW_STATS_RECENT		0 +#define WINDOW_STATS_MAX		1 +#define WINDOW_STATS_MAX_RECENT_AVG	2 +#define WINDOW_STATS_AVG		3 +#define WINDOW_STATS_INVALID_POLICY	4 + +#define SCHED_UPMIGRATE_MIN_NICE 15 +#define EXITING_TASK_MARKER	0xdeaddead + +#define UP_MIGRATION		1 +#define DOWN_MIGRATION		2 +#define IRQLOAD_MIGRATION	3 + +extern struct mutex policy_mutex; +extern unsigned int sched_ravg_window; +extern unsigned int sched_disable_window_stats; +extern unsigned int max_possible_freq; +extern unsigned int min_max_freq; +extern unsigned int pct_task_load(struct task_struct *p); +extern unsigned int max_possible_efficiency; +extern unsigned int min_possible_efficiency; +extern unsigned int max_capacity; +extern unsigned int min_capacity; +extern unsigned int max_load_scale_factor; +extern unsigned int max_possible_capacity; +extern unsigned int min_max_possible_capacity; +extern unsigned int max_power_cost; +extern unsigned int sched_init_task_load_windows; +extern unsigned int up_down_migrate_scale_factor; +extern unsigned int sysctl_sched_restrict_cluster_spill; +extern unsigned int sched_pred_alert_load; +extern struct sched_cluster init_cluster; +extern unsigned int  __read_mostly sched_short_sleep_task_threshold; +extern unsigned int  __read_mostly sched_long_cpu_selection_threshold; +extern unsigned int  __read_mostly sched_big_waker_task_load; +extern unsigned int  __read_mostly sched_small_wakee_task_load; +extern unsigned int  __read_mostly sched_spill_load; +extern unsigned int  __read_mostly sched_upmigrate; +extern unsigned int  __read_mostly sched_downmigrate; +extern unsigned int  __read_mostly sysctl_sched_spill_nr_run; +extern unsigned int  __read_mostly sched_load_granule; + +extern void init_new_task_load(struct task_struct *p, bool idle_task); +extern u64 sched_ktime_clock(void); +extern int got_boost_kick(void); +extern int register_cpu_cycle_counter_cb(struct cpu_cycle_counter_cb *cb); +extern void update_task_ravg(struct task_struct *p, struct rq *rq, int event, +						u64 wallclock, u64 irqtime); +extern bool early_detection_notify(struct rq *rq, u64 wallclock); +extern void clear_ed_task(struct task_struct *p, struct rq *rq); +extern void fixup_busy_time(struct task_struct *p, int new_cpu); +extern void clear_boost_kick(int cpu); +extern void clear_hmp_request(int cpu); +extern void mark_task_starting(struct task_struct *p); +extern void set_window_start(struct rq *rq); +extern void update_cluster_topology(void); +extern void note_task_waking(struct task_struct *p, u64 wallclock); +extern void set_task_last_switch_out(struct task_struct *p, u64 wallclock); +extern void init_clusters(void); +extern void reset_cpu_hmp_stats(int cpu, int reset_cra); +extern unsigned int max_task_load(void); +extern void sched_account_irqtime(int cpu, struct task_struct *curr, +				 u64 delta, u64 wallclock); +extern void sched_account_irqstart(int cpu, struct task_struct *curr, +				   u64 wallclock); +extern unsigned int cpu_temp(int cpu); +extern unsigned int nr_eligible_big_tasks(int cpu); +extern int update_preferred_cluster(struct related_thread_group *grp, +			struct task_struct *p, u32 old_load); +extern void set_preferred_cluster(struct related_thread_group *grp); +extern void add_new_task_to_grp(struct task_struct *new); +extern unsigned int update_freq_aggregate_threshold(unsigned int threshold); +extern void update_avg_burst(struct task_struct *p); +extern void update_avg(u64 *avg, u64 sample); + +#define NO_BOOST 0 +#define FULL_THROTTLE_BOOST 1 +#define CONSERVATIVE_BOOST 2 +#define RESTRAINED_BOOST 3 + +static inline struct sched_cluster *cpu_cluster(int cpu) +{ +	return cpu_rq(cpu)->cluster; +} + +static inline int cpu_capacity(int cpu) +{ +	return cpu_rq(cpu)->cluster->capacity; +} + +static inline int cpu_max_possible_capacity(int cpu) +{ +	return cpu_rq(cpu)->cluster->max_possible_capacity; +} + +static inline int cpu_load_scale_factor(int cpu) +{ +	return cpu_rq(cpu)->cluster->load_scale_factor; +} + +static inline int cpu_efficiency(int cpu) +{ +	return cpu_rq(cpu)->cluster->efficiency; +} + +static inline unsigned int cpu_cur_freq(int cpu) +{ +	return cpu_rq(cpu)->cluster->cur_freq; +} + +static inline unsigned int cpu_min_freq(int cpu) +{ +	return cpu_rq(cpu)->cluster->min_freq; +} + +static inline unsigned int cluster_max_freq(struct sched_cluster *cluster) +{ +	/* +	 * Governor and thermal driver don't know the other party's mitigation +	 * voting. So struct cluster saves both and return min() for current +	 * cluster fmax. +	 */ +	return min(cluster->max_mitigated_freq, cluster->max_freq); +} + +static inline unsigned int cpu_max_freq(int cpu) +{ +	return cluster_max_freq(cpu_rq(cpu)->cluster); +} + +static inline unsigned int cpu_max_possible_freq(int cpu) +{ +	return cpu_rq(cpu)->cluster->max_possible_freq; +} + +static inline int same_cluster(int src_cpu, int dst_cpu) +{ +	return cpu_rq(src_cpu)->cluster == cpu_rq(dst_cpu)->cluster; +} + +static inline int cpu_max_power_cost(int cpu) +{ +	return cpu_rq(cpu)->cluster->max_power_cost; +} + +static inline int cpu_min_power_cost(int cpu) +{ +	return cpu_rq(cpu)->cluster->min_power_cost; +} + +static inline u32 cpu_cycles_to_freq(u64 cycles, u64 period) +{ +	return div64_u64(cycles, period); +} + +static inline bool hmp_capable(void) +{ +	return max_possible_capacity != min_max_possible_capacity; +} + +static inline bool is_max_capacity_cpu(int cpu) +{ +	return cpu_max_possible_capacity(cpu) == max_possible_capacity; +} + +/* + * 'load' is in reference to "best cpu" at its best frequency. + * Scale that in reference to a given cpu, accounting for how bad it is + * in reference to "best cpu". + */ +static inline u64 scale_load_to_cpu(u64 task_load, int cpu) +{ +	u64 lsf = cpu_load_scale_factor(cpu); + +	if (lsf != 1024) { +		task_load *= lsf; +		task_load /= 1024; +	} + +	return task_load; +} + +static inline unsigned int task_load(struct task_struct *p) +{ +	return p->ravg.demand; +} + +static inline void +inc_cumulative_runnable_avg(struct hmp_sched_stats *stats, +				 struct task_struct *p) +{ +	u32 task_load; + +	if (sched_disable_window_stats) +		return; + +	task_load = sched_disable_window_stats ? 0 : p->ravg.demand; + +	stats->cumulative_runnable_avg += task_load; +	stats->pred_demands_sum += p->ravg.pred_demand; +} + +static inline void +dec_cumulative_runnable_avg(struct hmp_sched_stats *stats, +				struct task_struct *p) +{ +	u32 task_load; + +	if (sched_disable_window_stats) +		return; + +	task_load = sched_disable_window_stats ? 0 : p->ravg.demand; + +	stats->cumulative_runnable_avg -= task_load; + +	BUG_ON((s64)stats->cumulative_runnable_avg < 0); + +	stats->pred_demands_sum -= p->ravg.pred_demand; +	BUG_ON((s64)stats->pred_demands_sum < 0); +} + +static inline void +fixup_cumulative_runnable_avg(struct hmp_sched_stats *stats, +			      struct task_struct *p, s64 task_load_delta, +			      s64 pred_demand_delta) +{ +	if (sched_disable_window_stats) +		return; + +	stats->cumulative_runnable_avg += task_load_delta; +	BUG_ON((s64)stats->cumulative_runnable_avg < 0); + +	stats->pred_demands_sum += pred_demand_delta; +	BUG_ON((s64)stats->pred_demands_sum < 0); +} + +#define pct_to_real(tunable)	\ +		(div64_u64((u64)tunable * (u64)max_task_load(), 100)) + +#define real_to_pct(tunable)	\ +		(div64_u64((u64)tunable * (u64)100, (u64)max_task_load())) + +#define SCHED_HIGH_IRQ_TIMEOUT 3 +static inline u64 sched_irqload(int cpu) +{ +	struct rq *rq = cpu_rq(cpu); +	s64 delta; + +	delta = get_jiffies_64() - rq->irqload_ts; +	/* +	 * Current context can be preempted by irq and rq->irqload_ts can be +	 * updated by irq context so that delta can be negative. +	 * But this is okay and we can safely return as this means there +	 * was recent irq occurrence. +	 */ + +	if (delta < SCHED_HIGH_IRQ_TIMEOUT) +		return rq->avg_irqload; +	else +		return 0; +} + +static inline int sched_cpu_high_irqload(int cpu) +{ +	return sched_irqload(cpu) >= sysctl_sched_cpu_high_irqload; +} + +static inline bool task_in_related_thread_group(struct task_struct *p) +{ +	return !!(rcu_access_pointer(p->grp) != NULL); +} + +static inline +struct related_thread_group *task_related_thread_group(struct task_struct *p) +{ +	return rcu_dereference(p->grp); +} + +#define PRED_DEMAND_DELTA ((s64)new_pred_demand - p->ravg.pred_demand) + +extern void +check_for_freq_change(struct rq *rq, bool check_pred, bool check_groups); + +extern void notify_migration(int src_cpu, int dest_cpu, +			bool src_cpu_dead, struct task_struct *p); + +/* Is frequency of two cpus synchronized with each other? */ +static inline int same_freq_domain(int src_cpu, int dst_cpu) +{ +	struct rq *rq = cpu_rq(src_cpu); + +	if (src_cpu == dst_cpu) +		return 1; + +	return cpumask_test_cpu(dst_cpu, &rq->freq_domain_cpumask); +} + +#define	BOOST_KICK	0 +#define	CPU_RESERVED	1 + +static inline int is_reserved(int cpu) +{ +	struct rq *rq = cpu_rq(cpu); + +	return test_bit(CPU_RESERVED, &rq->hmp_flags); +} + +static inline int mark_reserved(int cpu) +{ +	struct rq *rq = cpu_rq(cpu); + +	/* Name boost_flags as hmp_flags? */ +	return test_and_set_bit(CPU_RESERVED, &rq->hmp_flags); +} + +static inline void clear_reserved(int cpu) +{ +	struct rq *rq = cpu_rq(cpu); + +	clear_bit(CPU_RESERVED, &rq->hmp_flags); +} + +static inline u64 cpu_cravg_sync(int cpu, int sync) +{ +	struct rq *rq = cpu_rq(cpu); +	u64 load; + +	load = rq->hmp_stats.cumulative_runnable_avg; + +	/* +	 * If load is being checked in a sync wakeup environment, +	 * we may want to discount the load of the currently running +	 * task. +	 */ +	if (sync && cpu == smp_processor_id()) { +		if (load > rq->curr->ravg.demand) +			load -= rq->curr->ravg.demand; +		else +			load = 0; +	} + +	return load; +} + +static inline bool is_short_burst_task(struct task_struct *p) +{ +	return p->ravg.avg_burst < sysctl_sched_short_burst && +	       p->ravg.avg_sleep_time > sysctl_sched_short_sleep; +} + +extern void check_for_migration(struct rq *rq, struct task_struct *p); +extern void pre_big_task_count_change(const struct cpumask *cpus); +extern void post_big_task_count_change(const struct cpumask *cpus); +extern void set_hmp_defaults(void); +extern int power_delta_exceeded(unsigned int cpu_cost, unsigned int base_cost); +extern unsigned int power_cost(int cpu, u64 demand); +extern void reset_all_window_stats(u64 window_start, unsigned int window_size); +extern int sched_boost(void); +extern int task_load_will_fit(struct task_struct *p, u64 task_load, int cpu, +					enum sched_boost_policy boost_policy); +extern enum sched_boost_policy sched_boost_policy(void); +extern int task_will_fit(struct task_struct *p, int cpu); +extern u64 cpu_load(int cpu); +extern u64 cpu_load_sync(int cpu, int sync); +extern int preferred_cluster(struct sched_cluster *cluster, +						struct task_struct *p); +extern void inc_nr_big_task(struct hmp_sched_stats *stats, +					struct task_struct *p); +extern void dec_nr_big_task(struct hmp_sched_stats *stats, +					struct task_struct *p); +extern void inc_rq_hmp_stats(struct rq *rq, +				struct task_struct *p, int change_cra); +extern void dec_rq_hmp_stats(struct rq *rq, +				struct task_struct *p, int change_cra); +extern void reset_hmp_stats(struct hmp_sched_stats *stats, int reset_cra); +extern int is_big_task(struct task_struct *p); +extern int upmigrate_discouraged(struct task_struct *p); +extern struct sched_cluster *rq_cluster(struct rq *rq); +extern int nr_big_tasks(struct rq *rq); +extern void fixup_nr_big_tasks(struct hmp_sched_stats *stats, +					struct task_struct *p, s64 delta); +extern void reset_task_stats(struct task_struct *p); +extern void reset_cfs_rq_hmp_stats(int cpu, int reset_cra); +extern void _inc_hmp_sched_stats_fair(struct rq *rq, +			struct task_struct *p, int change_cra); +extern u64 cpu_upmigrate_discourage_read_u64(struct cgroup_subsys_state *css, +					struct cftype *cft); +extern int cpu_upmigrate_discourage_write_u64(struct cgroup_subsys_state *css, +				struct cftype *cft, u64 upmigrate_discourage); +extern void sched_boost_parse_dt(void); +extern void clear_top_tasks_bitmap(unsigned long *bitmap); + +#if defined(CONFIG_SCHED_TUNE) && defined(CONFIG_CGROUP_SCHEDTUNE) +extern bool task_sched_boost(struct task_struct *p); +extern int sync_cgroup_colocation(struct task_struct *p, bool insert); +extern bool same_schedtune(struct task_struct *tsk1, struct task_struct *tsk2); +extern void update_cgroup_boost_settings(void); +extern void restore_cgroup_boost_settings(void); + +#else +static inline bool +same_schedtune(struct task_struct *tsk1, struct task_struct *tsk2) +{ +	return true; +} + +static inline bool task_sched_boost(struct task_struct *p) +{ +	return true; +} + +static inline void update_cgroup_boost_settings(void) { } +static inline void restore_cgroup_boost_settings(void) { } +#endif + +extern int alloc_related_thread_groups(void); + +#else	/* CONFIG_SCHED_HMP */ + +struct hmp_sched_stats; +struct related_thread_group; +struct sched_cluster; + +static inline enum sched_boost_policy sched_boost_policy(void) +{ +	return SCHED_BOOST_NONE; +} + +static inline bool task_sched_boost(struct task_struct *p) +{ +	return true; +} + +static inline int got_boost_kick(void) +{ +	return 0; +} + +static inline void update_task_ravg(struct task_struct *p, struct rq *rq, +				int event, u64 wallclock, u64 irqtime) { } + +static inline bool early_detection_notify(struct rq *rq, u64 wallclock) +{ +	return 0; +} + +static inline void clear_ed_task(struct task_struct *p, struct rq *rq) { } +static inline void fixup_busy_time(struct task_struct *p, int new_cpu) { } +static inline void clear_boost_kick(int cpu) { } +static inline void clear_hmp_request(int cpu) { } +static inline void mark_task_starting(struct task_struct *p) { } +static inline void set_window_start(struct rq *rq) { } +static inline void init_clusters(void) {} +static inline void update_cluster_topology(void) { } +static inline void note_task_waking(struct task_struct *p, u64 wallclock) { } +static inline void set_task_last_switch_out(struct task_struct *p, +					    u64 wallclock) { } + +static inline int task_will_fit(struct task_struct *p, int cpu) +{ +	return 1; +} + +static inline int select_best_cpu(struct task_struct *p, int target, +				  int reason, int sync) +{ +	return 0; +} + +static inline unsigned int power_cost(int cpu, u64 demand) +{ +	return SCHED_CAPACITY_SCALE; +} + +static inline int sched_boost(void) +{ +	return 0; +} + +static inline int is_big_task(struct task_struct *p) +{ +	return 0; +} + +static inline int nr_big_tasks(struct rq *rq) +{ +	return 0; +} + +static inline int is_cpu_throttling_imminent(int cpu) +{ +	return 0; +} + +static inline int is_task_migration_throttled(struct task_struct *p) +{ +	return 0; +} + +static inline unsigned int cpu_temp(int cpu) +{ +	return 0; +} + +static inline void +inc_rq_hmp_stats(struct rq *rq, struct task_struct *p, int change_cra) { } + +static inline void +dec_rq_hmp_stats(struct rq *rq, struct task_struct *p, int change_cra) { } + +static inline void +inc_hmp_sched_stats_fair(struct rq *rq, struct task_struct *p) { } + +static inline void +dec_hmp_sched_stats_fair(struct rq *rq, struct task_struct *p) { } + +static inline int +preferred_cluster(struct sched_cluster *cluster, struct task_struct *p) +{ +	return 1; +} + +static inline struct sched_cluster *rq_cluster(struct rq *rq) +{ +	return NULL; +} + +static inline void init_new_task_load(struct task_struct *p, bool idle_task) +{ +} + +static inline u64 scale_load_to_cpu(u64 load, int cpu) +{ +	return load; +} + +static inline unsigned int nr_eligible_big_tasks(int cpu) +{ +	return 0; +} + +static inline bool is_max_capacity_cpu(int cpu) { return true; } + +static inline int pct_task_load(struct task_struct *p) { return 0; } + +static inline int cpu_capacity(int cpu) +{ +	return SCHED_LOAD_SCALE; +} + +static inline int same_cluster(int src_cpu, int dst_cpu) { return 1; } + +static inline void inc_cumulative_runnable_avg(struct hmp_sched_stats *stats, +		 struct task_struct *p) +{ +} + +static inline void dec_cumulative_runnable_avg(struct hmp_sched_stats *stats, +		 struct task_struct *p) +{ +} + +static inline void sched_account_irqtime(int cpu, struct task_struct *curr, +				 u64 delta, u64 wallclock) +{ +} + +static inline void sched_account_irqstart(int cpu, struct task_struct *curr, +					  u64 wallclock) +{ +} + +static inline int sched_cpu_high_irqload(int cpu) { return 0; } + +static inline void set_preferred_cluster(struct related_thread_group *grp) { } + +static inline bool task_in_related_thread_group(struct task_struct *p) +{ +	return false; +} + +static inline +struct related_thread_group *task_related_thread_group(struct task_struct *p) +{ +	return NULL; +} + +static inline u32 task_load(struct task_struct *p) { return 0; } + +static inline int update_preferred_cluster(struct related_thread_group *grp, +			 struct task_struct *p, u32 old_load) +{ +	return 0; +} + +static inline void add_new_task_to_grp(struct task_struct *new) {} + +#define PRED_DEMAND_DELTA (0) + +static inline void +check_for_freq_change(struct rq *rq, bool check_pred, bool check_groups) { } + +static inline void notify_migration(int src_cpu, int dest_cpu, +			bool src_cpu_dead, struct task_struct *p) { } + +static inline int same_freq_domain(int src_cpu, int dst_cpu) +{ +	return 1; +} + +static inline void check_for_migration(struct rq *rq, struct task_struct *p) { } +static inline void pre_big_task_count_change(void) { } +static inline void post_big_task_count_change(void) { } +static inline void set_hmp_defaults(void) { } + +static inline void clear_reserved(int cpu) { } +static inline void sched_boost_parse_dt(void) {} +static inline int alloc_related_thread_groups(void) { return 0; } + +#define trace_sched_cpu_load(...) +#define trace_sched_cpu_load_lb(...) +#define trace_sched_cpu_load_cgroup(...) +#define trace_sched_cpu_load_wakeup(...) + +static inline void update_avg_burst(struct task_struct *p) {} + +#endif	/* CONFIG_SCHED_HMP */ + +/* + * Returns the rq capacity of any rq in a group. This does not play + * well with groups where rq capacity can change independently. + */ +#define group_rq_capacity(group) cpu_capacity(group_first_cpu(group)) +  #ifdef CONFIG_CGROUP_SCHED  /* @@ -1027,7 +1787,6 @@ static inline struct task_group *task_group(struct task_struct *p)  {  	return NULL;  } -  #endif /* CONFIG_CGROUP_SCHED */  static inline void __set_task_cpu(struct task_struct *p, unsigned int cpu) @@ -1181,6 +1940,7 @@ static inline void finish_lock_switch(struct rq *rq, struct task_struct *prev)  #define WF_SYNC		0x01		/* waker goes to sleep after wakeup */  #define WF_FORK		0x02		/* child wakeup after fork */  #define WF_MIGRATED	0x4		/* internal use, task got migrated */ +#define WF_NO_NOTIFIER	0x08		/* do not notify governor */  /*   * To aid in avoiding the subversion of "niceness" due to uneven distribution @@ -1235,19 +1995,41 @@ static const u32 prio_to_wmult[40] = {   /*  15 */ 119304647, 148102320, 186737708, 238609294, 286331153,  }; +/* + * {de,en}queue flags: + * + * DEQUEUE_SLEEP  - task is no longer runnable + * ENQUEUE_WAKEUP - task just became runnable + * + * SAVE/RESTORE - an otherwise spurious dequeue/enqueue, done to ensure tasks + *                are in a known state which allows modification. Such pairs + *                should preserve as much state as possible. + * + * MOVE - paired with SAVE/RESTORE, explicitly does not preserve the location + *        in the runqueue. + * + * ENQUEUE_HEAD      - place at front of runqueue (tail if not specified) + * ENQUEUE_REPLENISH - CBS (replenish runtime and postpone deadline) + * ENQUEUE_WAKING    - sched_class::task_waking was called + * + */ + +#define DEQUEUE_SLEEP		0x01 +#define DEQUEUE_SAVE		0x02 /* matches ENQUEUE_RESTORE */ +#define DEQUEUE_MOVE		0x04 /* matches ENQUEUE_MOVE */ +  #define ENQUEUE_WAKEUP		0x01 -#define ENQUEUE_HEAD		0x02 +#define ENQUEUE_RESTORE		0x02 +#define ENQUEUE_MOVE		0x04 + +#define ENQUEUE_HEAD		0x08 +#define ENQUEUE_REPLENISH	0x10  #ifdef CONFIG_SMP -#define ENQUEUE_WAKING		0x04	/* sched_class::task_waking was called */ +#define ENQUEUE_WAKING		0x20  #else  #define ENQUEUE_WAKING		0x00  #endif -#define ENQUEUE_REPLENISH	0x08 -#define ENQUEUE_RESTORE	0x10 -#define ENQUEUE_WAKEUP_NEW	0x20 - -#define DEQUEUE_SLEEP		0x01 -#define DEQUEUE_SAVE		0x02 +#define ENQUEUE_WAKEUP_NEW	0x40  #define RETRY_TASK		((void *)-1UL) @@ -1314,6 +2096,12 @@ struct sched_class {  #ifdef CONFIG_FAIR_GROUP_SCHED  	void (*task_change_group)(struct task_struct *p, int type);  #endif +#ifdef CONFIG_SCHED_HMP +	void (*inc_hmp_sched_stats)(struct rq *rq, struct task_struct *p); +	void (*dec_hmp_sched_stats)(struct rq *rq, struct task_struct *p); +	void (*fixup_hmp_sched_stats)(struct rq *rq, struct task_struct *p, +				      u32 new_task_load, u32 new_pred_demand); +#endif  };  static inline void put_prev_task(struct rq *rq, struct task_struct *prev) @@ -1338,6 +2126,7 @@ extern void init_max_cpu_capacity(struct max_cpu_capacity *mcc);  extern void update_group_capacity(struct sched_domain *sd, int cpu);  extern void trigger_load_balance(struct rq *rq); +extern void nohz_balance_clear_nohz_mask(int cpu);  extern void idle_enter_fair(struct rq *this_rq);  extern void idle_exit_fair(struct rq *this_rq); @@ -1395,7 +2184,9 @@ static inline int idle_get_state_idx(struct rq *rq)  }  #endif +#ifdef CONFIG_SYSRQ_SCHED_DEBUG  extern void sysrq_sched_debug_show(void); +#endif  extern void sched_init_granularity(void);  extern void update_max_interval(void); @@ -1422,6 +2213,7 @@ static inline void __add_nr_running(struct rq *rq, unsigned count)  {  	unsigned prev_nr = rq->nr_running; +	sched_update_nr_prod(cpu_of(rq), count, true);  	rq->nr_running = prev_nr + count;  	if (prev_nr < 2 && rq->nr_running >= 2) { @@ -1448,6 +2240,7 @@ static inline void __add_nr_running(struct rq *rq, unsigned count)  static inline void __sub_nr_running(struct rq *rq, unsigned count)  { +	sched_update_nr_prod(cpu_of(rq), count, false);  	rq->nr_running -= count;  } @@ -1611,6 +2404,7 @@ static inline unsigned long __cpu_util(int cpu, int delta)  		util = div64_u64(cpu_rq(cpu)->cumulative_runnable_avg,  				 walt_ravg_window >> SCHED_LOAD_SHIFT);  #endif +  	delta += util;  	if (delta < 0)  		return 0; @@ -1638,6 +2432,20 @@ static inline unsigned long cpu_util_freq(int cpu)  #endif +#ifdef CONFIG_SCHED_HMP +/* + * HMP and EAS are orthogonal. Hopefully the compiler just elides out all code + * with the energy_aware() check, so that we don't even pay the comparison + * penalty at runtime. + */ +#define energy_aware() false +#else +static inline bool energy_aware(void) +{ +	return sched_feat(ENERGY_AWARE); +} +#endif +  static inline void sched_rt_avg_update(struct rq *rq, u64 rt_delta)  {  	rq->rt_avg += rt_delta * arch_scale_freq_capacity(NULL, cpu_of(rq)); @@ -1878,6 +2686,11 @@ static inline void double_rq_unlock(struct rq *rq1, struct rq *rq2)  		__release(rq2->lock);  } +/* + * task_may_not_preempt - check whether a task may not be preemptible soon + */ +extern bool task_may_not_preempt(struct task_struct *task, int cpu); +  #else /* CONFIG_SMP */  /* @@ -1945,6 +2758,9 @@ enum rq_nohz_flag_bits {  	NOHZ_BALANCE_KICK,  }; +#define NOHZ_KICK_ANY 0 +#define NOHZ_KICK_RESTRICT 1 +  #define nohz_flags(cpu)	(&cpu_rq(cpu)->nohz_flags)  #endif | 
