Files
android_kernel_samsung_sm8750/include/linux/sched/walt.h
2025-08-11 14:29:00 +02:00

301 lines
7.6 KiB
C

/* SPDX-License-Identifier: GPL-2.0-only */
/*
* Copyright (c) 2021, The Linux Foundation. All rights reserved.
* Copyright (c) 2022-2024, Qualcomm Innovation Center, Inc. All rights reserved.
*/
#ifndef _LINUX_SCHED_WALT_H
#define _LINUX_SCHED_WALT_H
#include <linux/types.h>
#include <linux/spinlock_types.h>
#include <linux/cpumask.h>
enum pause_client {
PAUSE_CORE_CTL = 0x01,
PAUSE_THERMAL = 0x02,
PAUSE_HYP = 0x04,
PAUSE_SBT = 0x08,
};
#define NO_BOOST 0
#define FULL_THROTTLE_BOOST 1
#define CONSERVATIVE_BOOST 2
#define RESTRAINED_BOOST 3
#define STORAGE_BOOST 4
#define BALANCE_BOOST 5
#define FULL_THROTTLE_BOOST_DISABLE -1
#define CONSERVATIVE_BOOST_DISABLE -2
#define RESTRAINED_BOOST_DISABLE -3
#define STORAGE_BOOST_DISABLE -4
#define BALANCE_BOOST_DISABLE -5
#define MAX_NUM_BOOST_TYPE (BALANCE_BOOST+1)
#if IS_ENABLED(CONFIG_SCHED_WALT)
#define MAX_CPUS_PER_CLUSTER 6
#define MAX_CLUSTERS 4
struct core_ctl_notif_data {
unsigned int nr_big;
unsigned int coloc_load_pct;
unsigned int ta_util_pct[MAX_CLUSTERS];
unsigned int cur_cap_pct[MAX_CLUSTERS];
};
enum task_boost_type {
TASK_BOOST_NONE = 0,
TASK_BOOST_ON_MID,
TASK_BOOST_ON_MAX,
TASK_BOOST_STRICT_MAX,
TASK_BOOST_END,
};
#define WALT_NR_CPUS 8
#define RAVG_HIST_SIZE 5
/* wts->bucket_bitmask needs to be updated if NUM_BUSY_BUCKETS > 16 */
#define NUM_BUSY_BUCKETS 16
#define NUM_BUSY_BUCKETS_SHIFT 4
struct walt_related_thread_group {
int id;
raw_spinlock_t lock;
struct list_head tasks;
struct list_head list;
bool skip_min;
struct rcu_head rcu;
u64 last_update;
u64 downmigrate_ts;
u64 start_ktime_ts;
};
struct walt_task_struct {
/*
* 'mark_start' marks the beginning of an event (task waking up, task
* starting to execute, task being preempted) within a window
*
* 'sum' represents how runnable a task has been within current
* window. It incorporates both running time and wait time and is
* frequency scaled.
*
* 'sum_history' keeps track of history of 'sum' seen over previous
* RAVG_HIST_SIZE windows. Windows where task was entirely sleeping are
* ignored.
*
* 'demand' represents maximum sum seen over previous
* sysctl_sched_ravg_hist_size windows. 'demand' could drive frequency
* demand for tasks.
*
* 'curr_window_cpu' represents task's contribution to cpu busy time on
* various CPUs in the current window
*
* 'prev_window_cpu' represents task's contribution to cpu busy time on
* various CPUs in the previous window
*
* 'curr_window' represents the sum of all entries in curr_window_cpu
*
* 'prev_window' represents the sum of all entries in prev_window_cpu
*
* 'pred_demand_scaled' represents task's current predicted cpu busy time
* in terms of 1024 units
*
* 'busy_buckets' groups historical busy time into different buckets
* used for prediction
*
* 'demand_scaled' represents task's demand scaled to 1024
*
* 'prev_on_rq' tracks enqueue/dequeue of a task for error conditions
* 0 = nothing, 1 = enqueued, 2 = dequeued
*/
u32 flags;
u64 mark_start;
u64 window_start;
u32 sum, demand;
u32 coloc_demand;
u32 sum_history[RAVG_HIST_SIZE];
u16 sum_history_util[RAVG_HIST_SIZE];
u32 curr_window_cpu[WALT_NR_CPUS];
u32 prev_window_cpu[WALT_NR_CPUS];
u32 curr_window, prev_window;
u8 busy_buckets[NUM_BUSY_BUCKETS];
u16 bucket_bitmask;
u16 demand_scaled;
u16 pred_demand_scaled;
u64 active_time;
u64 last_win_size;
int boost;
bool wake_up_idle;
bool misfit;
bool rtg_high_prio;
u8 low_latency;
u64 boost_period;
u64 boost_expires;
u64 last_sleep_ts;
u32 init_load_pct;
u32 unfilter;
u64 last_wake_ts;
u64 last_enqueued_ts;
struct walt_related_thread_group __rcu *grp;
struct list_head grp_list;
u64 cpu_cycles;
bool iowaited;
int prev_on_rq;
int prev_on_rq_cpu;
struct list_head mvp_list;
u64 sum_exec_snapshot_for_slice;
u64 sum_exec_snapshot_for_total;
u64 total_exec;
int mvp_prio;
int cidx;
int load_boost;
int64_t boosted_task_load;
int prev_cpu;
int new_cpu;
u8 enqueue_after_migration;
u8 hung_detect_status;
int pipeline_cpu;
cpumask_t reduce_mask;
u64 mark_start_birth_ts;
u8 high_util_history;
u8 mpam_part_id;
u8 yield_state;
u16 busy_bitmap;
u32 period_contrib_run;
};
#define wts_to_ts(wts) ({ \
void *__mptr = (void *)(wts); \
((struct task_struct *)(__mptr - \
offsetof(struct task_struct, android_vendor_data1))); })
static inline bool sched_get_wake_up_idle(struct task_struct *p)
{
struct walt_task_struct *wts = (struct walt_task_struct *) p->android_vendor_data1;
return wts->wake_up_idle;
}
static inline int sched_set_wake_up_idle(struct task_struct *p, bool wake_up_idle)
{
struct walt_task_struct *wts = (struct walt_task_struct *) p->android_vendor_data1;
wts->wake_up_idle = wake_up_idle;
return 0;
}
static inline void set_wake_up_idle(bool wake_up_idle)
{
struct walt_task_struct *wts = (struct walt_task_struct *) current->android_vendor_data1;
wts->wake_up_idle = wake_up_idle;
}
extern int sched_lpm_disallowed_time(int cpu, u64 *timeout);
extern int set_task_boost(int boost, u64 period);
struct notifier_block;
extern void core_ctl_notifier_register(struct notifier_block *n);
extern void core_ctl_notifier_unregister(struct notifier_block *n);
extern int core_ctl_set_boost(bool boost);
extern int walt_set_cpus_taken(struct cpumask *set);
extern int walt_unset_cpus_taken(struct cpumask *unset);
extern cpumask_t walt_get_cpus_taken(void);
extern int walt_get_cpus_in_state1(struct cpumask *cpus);
extern void sched_walt_oscillate(unsigned int busy_cpu);
extern int walt_pause_cpus(struct cpumask *cpus, enum pause_client client);
extern int walt_resume_cpus(struct cpumask *cpus, enum pause_client client);
extern int walt_partial_pause_cpus(struct cpumask *cpus, enum pause_client client);
extern int walt_partial_resume_cpus(struct cpumask *cpus, enum pause_client client);
extern int sched_set_boost(int type);
extern bool should_boost_bus_dcvs(void);
extern cpumask_t walt_get_halted_cpus(void);
#else
static inline void sched_walt_oscillate(unsigned int busy_cpu) { }
static inline int sched_lpm_disallowed_time(int cpu, u64 *timeout)
{
return INT_MAX;
}
static inline int set_task_boost(int boost, u64 period)
{
return 0;
}
static inline bool sched_get_wake_up_idle(struct task_struct *p)
{
return false;
}
static inline int sched_set_wake_up_idle(struct task_struct *p, bool wake_up_idle)
{
return 0;
}
static inline void set_wake_up_idle(bool wake_up_idle)
{
}
static inline int core_ctl_set_boost(bool boost)
{
return 0;
}
static inline void core_ctl_notifier_register(struct notifier_block *n)
{
}
static inline void core_ctl_notifier_unregister(struct notifier_block *n)
{
}
static inline int walt_pause_cpus(struct cpumask *cpus, enum pause_client client)
{
return 0;
}
static inline int walt_resume_cpus(struct cpumask *cpus, enum pause_client client)
{
return 0;
}
inline int walt_partial_pause_cpus(struct cpumask *cpus, enum pause_client client)
{
return 0;
}
inline int walt_partial_resume_cpus(struct cpumask *cpus, enum pause_client client)
{
return 0;
}
static inline void walt_set_cpus_taken(struct cpumask *set)
{
}
static inline void walt_unset_cpus_taken(struct cpumask *unset)
{
}
static inline cpumask_t walt_get_cpus_taken(void)
{
cpumask_t t = { CPU_BITS_NONE };
return t;
}
static inline int sched_set_boost(int type)
{
return -EINVAL;
}
static inline bool should_boost_bus_dcvs(void)
{
return false;
}
static inline cpumask_t walt_get_halted_cpus(void)
{
cpumask_t t = { CPU_BITS_NONE };
return t;
}
#endif
#endif /* _LINUX_SCHED_WALT_H */