| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 1 | /* | 
|  | 2 | * cpuidle.h - a generic framework for CPU idle power management | 
|  | 3 | * | 
|  | 4 | * (C) 2007 Venkatesh Pallipadi <venkatesh.pallipadi@intel.com> | 
|  | 5 | *          Shaohua Li <shaohua.li@intel.com> | 
|  | 6 | *          Adam Belay <abelay@novell.com> | 
|  | 7 | * | 
|  | 8 | * This code is licenced under the GPL. | 
|  | 9 | */ | 
|  | 10 |  | 
|  | 11 | #ifndef _LINUX_CPUIDLE_H | 
|  | 12 | #define _LINUX_CPUIDLE_H | 
|  | 13 |  | 
|  | 14 | #include <linux/percpu.h> | 
|  | 15 | #include <linux/list.h> | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 16 | #include <linux/kobject.h> | 
|  | 17 | #include <linux/completion.h> | 
| Robert Lee | e168979 | 2012-03-20 15:22:42 -0500 | [diff] [blame] | 18 | #include <linux/hrtimer.h> | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 19 |  | 
|  | 20 | #define CPUIDLE_STATE_MAX	8 | 
|  | 21 | #define CPUIDLE_NAME_LEN	16 | 
| Venkatesh Pallipadi | 4fcb2fc | 2008-02-11 17:46:31 -0800 | [diff] [blame] | 22 | #define CPUIDLE_DESC_LEN	32 | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 23 |  | 
| Paul Gortmaker | de47725 | 2011-05-26 13:46:22 -0400 | [diff] [blame] | 24 | struct module; | 
|  | 25 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 26 | struct cpuidle_device; | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 27 | struct cpuidle_driver; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 28 |  | 
|  | 29 |  | 
|  | 30 | /**************************** | 
|  | 31 | * CPUIDLE DEVICE INTERFACE * | 
|  | 32 | ****************************/ | 
|  | 33 |  | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 34 | struct cpuidle_state_usage { | 
|  | 35 | void		*driver_data; | 
|  | 36 |  | 
| ShuoX Liu | dc7fd27 | 2012-07-03 19:05:31 +0200 | [diff] [blame] | 37 | unsigned long long	disable; | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 38 | unsigned long long	usage; | 
|  | 39 | unsigned long long	time; /* in US */ | 
|  | 40 | }; | 
|  | 41 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 42 | struct cpuidle_state { | 
|  | 43 | char		name[CPUIDLE_NAME_LEN]; | 
| Venkatesh Pallipadi | 4fcb2fc | 2008-02-11 17:46:31 -0800 | [diff] [blame] | 44 | char		desc[CPUIDLE_DESC_LEN]; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 45 |  | 
|  | 46 | unsigned int	flags; | 
|  | 47 | unsigned int	exit_latency; /* in US */ | 
| Boris Ostrovsky | 02401c0 | 2012-03-13 19:55:10 +0100 | [diff] [blame] | 48 | int		power_usage; /* in mW */ | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 49 | unsigned int	target_residency; /* in US */ | 
| Rafael J. Wysocki | cbc9ef0 | 2012-07-03 19:07:42 +0200 | [diff] [blame] | 50 | bool		disabled; /* disabled on all CPUs */ | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 51 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 52 | int (*enter)	(struct cpuidle_device *dev, | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 53 | struct cpuidle_driver *drv, | 
| Deepthi Dharwar | e978aa7 | 2011-10-28 16:20:09 +0530 | [diff] [blame] | 54 | int index); | 
| Boris Ostrovsky | 1a022e3 | 2012-03-13 19:55:09 +0100 | [diff] [blame] | 55 |  | 
|  | 56 | int (*enter_dead) (struct cpuidle_device *dev, int index); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 57 | }; | 
|  | 58 |  | 
|  | 59 | /* Idle State Flags */ | 
|  | 60 | #define CPUIDLE_FLAG_TIME_VALID	(0x01) /* is residency time measurable? */ | 
| Colin Cross | 4126c01 | 2012-05-07 17:57:41 -0700 | [diff] [blame] | 61 | #define CPUIDLE_FLAG_COUPLED	(0x02) /* state applies to multiple cpus */ | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 62 |  | 
|  | 63 | #define CPUIDLE_DRIVER_FLAGS_MASK (0xFFFF0000) | 
|  | 64 |  | 
|  | 65 | /** | 
|  | 66 | * cpuidle_get_statedata - retrieves private driver state data | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 67 | * @st_usage: the state usage statistics | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 68 | */ | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 69 | static inline void *cpuidle_get_statedata(struct cpuidle_state_usage *st_usage) | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 70 | { | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 71 | return st_usage->driver_data; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 72 | } | 
|  | 73 |  | 
|  | 74 | /** | 
|  | 75 | * cpuidle_set_statedata - stores private driver state data | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 76 | * @st_usage: the state usage statistics | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 77 | * @data: the private data | 
|  | 78 | */ | 
|  | 79 | static inline void | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 80 | cpuidle_set_statedata(struct cpuidle_state_usage *st_usage, void *data) | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 81 | { | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 82 | st_usage->driver_data = data; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 83 | } | 
|  | 84 |  | 
|  | 85 | struct cpuidle_state_kobj { | 
|  | 86 | struct cpuidle_state *state; | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 87 | struct cpuidle_state_usage *state_usage; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 88 | struct completion kobj_unregister; | 
|  | 89 | struct kobject kobj; | 
|  | 90 | }; | 
|  | 91 |  | 
|  | 92 | struct cpuidle_device { | 
| Venkatesh Pallipadi | dcb84f3 | 2008-05-19 19:09:27 -0400 | [diff] [blame] | 93 | unsigned int		registered:1; | 
| Harvey Harrison | b5556a6 | 2008-02-06 22:39:44 +0100 | [diff] [blame] | 94 | unsigned int		enabled:1; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 95 | unsigned int		cpu; | 
|  | 96 |  | 
|  | 97 | int			last_residency; | 
|  | 98 | int			state_count; | 
| Deepthi Dharwar | 4202735 | 2011-10-28 16:20:33 +0530 | [diff] [blame] | 99 | struct cpuidle_state_usage	states_usage[CPUIDLE_STATE_MAX]; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 100 | struct cpuidle_state_kobj *kobjs[CPUIDLE_STATE_MAX]; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 101 |  | 
|  | 102 | struct list_head 	device_list; | 
|  | 103 | struct kobject		kobj; | 
|  | 104 | struct completion	kobj_unregister; | 
| Colin Cross | 4126c01 | 2012-05-07 17:57:41 -0700 | [diff] [blame] | 105 |  | 
|  | 106 | #ifdef CONFIG_ARCH_NEEDS_CPU_IDLE_COUPLED | 
|  | 107 | int			safe_state_index; | 
|  | 108 | cpumask_t		coupled_cpus; | 
|  | 109 | struct cpuidle_coupled	*coupled; | 
|  | 110 | #endif | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 111 | }; | 
|  | 112 |  | 
|  | 113 | DECLARE_PER_CPU(struct cpuidle_device *, cpuidle_devices); | 
|  | 114 |  | 
|  | 115 | /** | 
|  | 116 | * cpuidle_get_last_residency - retrieves the last state's residency time | 
|  | 117 | * @dev: the target CPU | 
|  | 118 | * | 
|  | 119 | * NOTE: this value is invalid if CPUIDLE_FLAG_TIME_VALID isn't set | 
|  | 120 | */ | 
|  | 121 | static inline int cpuidle_get_last_residency(struct cpuidle_device *dev) | 
|  | 122 | { | 
|  | 123 | return dev->last_residency; | 
|  | 124 | } | 
|  | 125 |  | 
|  | 126 |  | 
|  | 127 | /**************************** | 
|  | 128 | * CPUIDLE DRIVER INTERFACE * | 
|  | 129 | ****************************/ | 
|  | 130 |  | 
|  | 131 | struct cpuidle_driver { | 
| Daniel Lezcano | db70b04 | 2012-03-26 14:51:27 +0200 | [diff] [blame] | 132 | const char		*name; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 133 | struct module 		*owner; | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 134 |  | 
|  | 135 | unsigned int		power_specified:1; | 
| Robert Lee | e168979 | 2012-03-20 15:22:42 -0500 | [diff] [blame] | 136 | /* set to 1 to use the core cpuidle time keeping (for all states). */ | 
|  | 137 | unsigned int		en_core_tk_irqen:1; | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 138 | struct cpuidle_state	states[CPUIDLE_STATE_MAX]; | 
|  | 139 | int			state_count; | 
|  | 140 | int			safe_state_index; | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 141 | }; | 
|  | 142 |  | 
|  | 143 | #ifdef CONFIG_CPU_IDLE | 
| Len Brown | d91ee58 | 2011-04-01 18:28:35 -0400 | [diff] [blame] | 144 | extern void disable_cpuidle(void); | 
| Len Brown | a0bfa13 | 2011-04-01 19:34:59 -0400 | [diff] [blame] | 145 | extern int cpuidle_idle_call(void); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 146 | extern int cpuidle_register_driver(struct cpuidle_driver *drv); | 
| Rafael J. Wysocki | 6e797a0 | 2012-06-16 15:20:11 +0200 | [diff] [blame] | 147 | extern struct cpuidle_driver *cpuidle_get_driver(void); | 
|  | 148 | extern struct cpuidle_driver *cpuidle_driver_ref(void); | 
|  | 149 | extern void cpuidle_driver_unref(void); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 150 | extern void cpuidle_unregister_driver(struct cpuidle_driver *drv); | 
|  | 151 | extern int cpuidle_register_device(struct cpuidle_device *dev); | 
|  | 152 | extern void cpuidle_unregister_device(struct cpuidle_device *dev); | 
|  | 153 |  | 
|  | 154 | extern void cpuidle_pause_and_lock(void); | 
|  | 155 | extern void cpuidle_resume_and_unlock(void); | 
| Preeti U Murthy | 8651f97 | 2012-07-09 10:12:56 +0200 | [diff] [blame] | 156 | extern void cpuidle_pause(void); | 
|  | 157 | extern void cpuidle_resume(void); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 158 | extern int cpuidle_enable_device(struct cpuidle_device *dev); | 
|  | 159 | extern void cpuidle_disable_device(struct cpuidle_device *dev); | 
| Robert Lee | e168979 | 2012-03-20 15:22:42 -0500 | [diff] [blame] | 160 | extern int cpuidle_wrap_enter(struct cpuidle_device *dev, | 
|  | 161 | struct cpuidle_driver *drv, int index, | 
|  | 162 | int (*enter)(struct cpuidle_device *dev, | 
|  | 163 | struct cpuidle_driver *drv, int index)); | 
| Boris Ostrovsky | 1a022e3 | 2012-03-13 19:55:09 +0100 | [diff] [blame] | 164 | extern int cpuidle_play_dead(void); | 
|  | 165 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 166 | #else | 
| Len Brown | d91ee58 | 2011-04-01 18:28:35 -0400 | [diff] [blame] | 167 | static inline void disable_cpuidle(void) { } | 
| Len Brown | a0bfa13 | 2011-04-01 19:34:59 -0400 | [diff] [blame] | 168 | static inline int cpuidle_idle_call(void) { return -ENODEV; } | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 169 | static inline int cpuidle_register_driver(struct cpuidle_driver *drv) | 
| Len Brown | 6b2c676 | 2010-05-11 16:50:52 -0400 | [diff] [blame] | 170 | {return -ENODEV; } | 
| Len Brown | 752138d | 2010-05-22 16:57:26 -0400 | [diff] [blame] | 171 | static inline struct cpuidle_driver *cpuidle_get_driver(void) {return NULL; } | 
| Rafael J. Wysocki | 6e797a0 | 2012-06-16 15:20:11 +0200 | [diff] [blame] | 172 | static inline struct cpuidle_driver *cpuidle_driver_ref(void) {return NULL; } | 
|  | 173 | static inline void cpuidle_driver_unref(void) {} | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 174 | static inline void cpuidle_unregister_driver(struct cpuidle_driver *drv) { } | 
|  | 175 | static inline int cpuidle_register_device(struct cpuidle_device *dev) | 
| Len Brown | 6b2c676 | 2010-05-11 16:50:52 -0400 | [diff] [blame] | 176 | {return -ENODEV; } | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 177 | static inline void cpuidle_unregister_device(struct cpuidle_device *dev) { } | 
|  | 178 |  | 
|  | 179 | static inline void cpuidle_pause_and_lock(void) { } | 
|  | 180 | static inline void cpuidle_resume_and_unlock(void) { } | 
| Preeti U Murthy | 8651f97 | 2012-07-09 10:12:56 +0200 | [diff] [blame] | 181 | static inline void cpuidle_pause(void) { } | 
|  | 182 | static inline void cpuidle_resume(void) { } | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 183 | static inline int cpuidle_enable_device(struct cpuidle_device *dev) | 
| Len Brown | 6b2c676 | 2010-05-11 16:50:52 -0400 | [diff] [blame] | 184 | {return -ENODEV; } | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 185 | static inline void cpuidle_disable_device(struct cpuidle_device *dev) { } | 
| Robert Lee | e168979 | 2012-03-20 15:22:42 -0500 | [diff] [blame] | 186 | static inline int cpuidle_wrap_enter(struct cpuidle_device *dev, | 
|  | 187 | struct cpuidle_driver *drv, int index, | 
|  | 188 | int (*enter)(struct cpuidle_device *dev, | 
|  | 189 | struct cpuidle_driver *drv, int index)) | 
|  | 190 | { return -ENODEV; } | 
| Boris Ostrovsky | 1a022e3 | 2012-03-13 19:55:09 +0100 | [diff] [blame] | 191 | static inline int cpuidle_play_dead(void) {return -ENODEV; } | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 192 |  | 
|  | 193 | #endif | 
|  | 194 |  | 
| Colin Cross | 20ff51a | 2012-05-07 17:57:42 -0700 | [diff] [blame] | 195 | #ifdef CONFIG_ARCH_NEEDS_CPU_IDLE_COUPLED | 
|  | 196 | void cpuidle_coupled_parallel_barrier(struct cpuidle_device *dev, atomic_t *a); | 
|  | 197 | #endif | 
|  | 198 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 199 | /****************************** | 
|  | 200 | * CPUIDLE GOVERNOR INTERFACE * | 
|  | 201 | ******************************/ | 
|  | 202 |  | 
|  | 203 | struct cpuidle_governor { | 
|  | 204 | char			name[CPUIDLE_NAME_LEN]; | 
|  | 205 | struct list_head 	governor_list; | 
|  | 206 | unsigned int		rating; | 
|  | 207 |  | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 208 | int  (*enable)		(struct cpuidle_driver *drv, | 
|  | 209 | struct cpuidle_device *dev); | 
|  | 210 | void (*disable)		(struct cpuidle_driver *drv, | 
|  | 211 | struct cpuidle_device *dev); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 212 |  | 
| Deepthi Dharwar | 46bcfad | 2011-10-28 16:20:42 +0530 | [diff] [blame] | 213 | int  (*select)		(struct cpuidle_driver *drv, | 
|  | 214 | struct cpuidle_device *dev); | 
| Deepthi Dharwar | e978aa7 | 2011-10-28 16:20:09 +0530 | [diff] [blame] | 215 | void (*reflect)		(struct cpuidle_device *dev, int index); | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 216 |  | 
|  | 217 | struct module 		*owner; | 
|  | 218 | }; | 
|  | 219 |  | 
|  | 220 | #ifdef CONFIG_CPU_IDLE | 
|  | 221 |  | 
|  | 222 | extern int cpuidle_register_governor(struct cpuidle_governor *gov); | 
|  | 223 | extern void cpuidle_unregister_governor(struct cpuidle_governor *gov); | 
|  | 224 |  | 
|  | 225 | #else | 
|  | 226 |  | 
|  | 227 | static inline int cpuidle_register_governor(struct cpuidle_governor *gov) | 
|  | 228 | {return 0;} | 
|  | 229 | static inline void cpuidle_unregister_governor(struct cpuidle_governor *gov) { } | 
|  | 230 |  | 
|  | 231 | #endif | 
|  | 232 |  | 
| venkatesh.pallipadi@intel.com | 9a0b841 | 2008-01-31 17:35:06 -0800 | [diff] [blame] | 233 | #ifdef CONFIG_ARCH_HAS_CPU_RELAX | 
|  | 234 | #define CPUIDLE_DRIVER_STATE_START	1 | 
|  | 235 | #else | 
|  | 236 | #define CPUIDLE_DRIVER_STATE_START	0 | 
|  | 237 | #endif | 
|  | 238 |  | 
| Len Brown | 4f86d3a | 2007-10-03 18:58:00 -0400 | [diff] [blame] | 239 | #endif /* _LINUX_CPUIDLE_H */ |