blob: c5d358837461bf4d65716dd7a41d14d7341891d5 [file] [log] [blame]
Jean Pihet91ff4cb2011-08-25 15:35:41 +02001/*
2 * Devices PM QoS constraints management
3 *
4 * Copyright (C) 2011 Texas Instruments, Inc.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 *
11 * This module exposes the interface to kernel space for specifying
12 * per-device PM QoS dependencies. It provides infrastructure for registration
13 * of:
14 *
15 * Dependents on a QoS value : register requests
16 * Watchers of QoS value : get notified when target QoS value changes
17 *
18 * This QoS design is best effort based. Dependents register their QoS needs.
19 * Watchers register to keep track of the current QoS needs of the system.
Jean Pihetb66213c2011-08-25 15:35:47 +020020 * Watchers can register different types of notification callbacks:
21 * . a per-device notification callback using the dev_pm_qos_*_notifier API.
22 * The notification chain data is stored in the per-device constraint
23 * data struct.
24 * . a system-wide notification callback using the dev_pm_qos_*_global_notifier
25 * API. The notification chain data is stored in a static variable.
Jean Pihet91ff4cb2011-08-25 15:35:41 +020026 *
27 * Note about the per-device constraint data struct allocation:
28 * . The per-device constraints data struct ptr is tored into the device
29 * dev_pm_info.
30 * . To minimize the data usage by the per-device constraints, the data struct
31 * is only allocated at the first call to dev_pm_qos_add_request.
32 * . The data is later free'd when the device is removed from the system.
Jean Pihet91ff4cb2011-08-25 15:35:41 +020033 * . A global mutex protects the constraints users from the data being
34 * allocated and free'd.
35 */
36
37#include <linux/pm_qos.h>
38#include <linux/spinlock.h>
39#include <linux/slab.h>
40#include <linux/device.h>
41#include <linux/mutex.h>
Paul Gortmaker1b6bc322011-05-27 07:12:15 -040042#include <linux/export.h>
Jean Pihet91ff4cb2011-08-25 15:35:41 +020043
44
45static DEFINE_MUTEX(dev_pm_qos_mtx);
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020046
Jean Pihetb66213c2011-08-25 15:35:47 +020047static BLOCKING_NOTIFIER_HEAD(dev_pm_notifiers);
48
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020049/**
Rafael J. Wysocki00dc9ad2011-12-01 00:01:31 +010050 * __dev_pm_qos_read_value - Get PM QoS constraint for a given device.
51 * @dev: Device to get the PM QoS constraint value for.
52 *
53 * This routine must be called with dev->power.lock held.
54 */
55s32 __dev_pm_qos_read_value(struct device *dev)
56{
57 struct pm_qos_constraints *c = dev->power.constraints;
58
59 return c ? pm_qos_read_value(c) : 0;
60}
61
62/**
63 * dev_pm_qos_read_value - Get PM QoS constraint for a given device (locked).
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020064 * @dev: Device to get the PM QoS constraint value for.
65 */
66s32 dev_pm_qos_read_value(struct device *dev)
67{
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020068 unsigned long flags;
Rafael J. Wysocki00dc9ad2011-12-01 00:01:31 +010069 s32 ret;
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020070
71 spin_lock_irqsave(&dev->power.lock, flags);
Rafael J. Wysocki00dc9ad2011-12-01 00:01:31 +010072 ret = __dev_pm_qos_read_value(dev);
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +020073 spin_unlock_irqrestore(&dev->power.lock, flags);
74
75 return ret;
76}
77
Jean Pihetb66213c2011-08-25 15:35:47 +020078/*
79 * apply_constraint
80 * @req: constraint request to apply
81 * @action: action to perform add/update/remove, of type enum pm_qos_req_action
82 * @value: defines the qos request
83 *
84 * Internal function to update the constraints list using the PM QoS core
85 * code and if needed call the per-device and the global notification
86 * callbacks
87 */
88static int apply_constraint(struct dev_pm_qos_request *req,
89 enum pm_qos_req_action action, int value)
90{
91 int ret, curr_value;
92
93 ret = pm_qos_update_target(req->dev->power.constraints,
94 &req->node, action, value);
95
96 if (ret) {
97 /* Call the global callbacks if needed */
98 curr_value = pm_qos_read_value(req->dev->power.constraints);
99 blocking_notifier_call_chain(&dev_pm_notifiers,
100 (unsigned long)curr_value,
101 req);
102 }
103
104 return ret;
105}
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200106
107/*
108 * dev_pm_qos_constraints_allocate
109 * @dev: device to allocate data for
110 *
111 * Called at the first call to add_request, for constraint data allocation
112 * Must be called with the dev_pm_qos_mtx mutex held
113 */
114static int dev_pm_qos_constraints_allocate(struct device *dev)
115{
116 struct pm_qos_constraints *c;
117 struct blocking_notifier_head *n;
118
119 c = kzalloc(sizeof(*c), GFP_KERNEL);
120 if (!c)
121 return -ENOMEM;
122
123 n = kzalloc(sizeof(*n), GFP_KERNEL);
124 if (!n) {
125 kfree(c);
126 return -ENOMEM;
127 }
128 BLOCKING_INIT_NOTIFIER_HEAD(n);
129
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200130 plist_head_init(&c->list);
131 c->target_value = PM_QOS_DEV_LAT_DEFAULT_VALUE;
132 c->default_value = PM_QOS_DEV_LAT_DEFAULT_VALUE;
133 c->type = PM_QOS_MIN;
134 c->notifiers = n;
135
136 spin_lock_irq(&dev->power.lock);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200137 dev->power.constraints = c;
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200138 spin_unlock_irq(&dev->power.lock);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200139
140 return 0;
141}
142
143/**
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200144 * dev_pm_qos_constraints_init - Initalize device's PM QoS constraints pointer.
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200145 * @dev: target device
146 *
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200147 * Called from the device PM subsystem during device insertion under
148 * device_pm_lock().
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200149 */
150void dev_pm_qos_constraints_init(struct device *dev)
151{
152 mutex_lock(&dev_pm_qos_mtx);
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200153 dev->power.constraints = NULL;
154 dev->power.power_state = PMSG_ON;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200155 mutex_unlock(&dev_pm_qos_mtx);
156}
157
158/**
159 * dev_pm_qos_constraints_destroy
160 * @dev: target device
161 *
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200162 * Called from the device PM subsystem on device removal under device_pm_lock().
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200163 */
164void dev_pm_qos_constraints_destroy(struct device *dev)
165{
166 struct dev_pm_qos_request *req, *tmp;
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200167 struct pm_qos_constraints *c;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200168
169 mutex_lock(&dev_pm_qos_mtx);
170
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200171 dev->power.power_state = PMSG_INVALID;
172 c = dev->power.constraints;
173 if (!c)
174 goto out;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200175
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200176 /* Flush the constraints list for the device */
177 plist_for_each_entry_safe(req, tmp, &c->list, node) {
178 /*
179 * Update constraints list and call the notification
180 * callbacks if needed
181 */
182 apply_constraint(req, PM_QOS_REMOVE_REQ, PM_QOS_DEFAULT_VALUE);
183 memset(req, 0, sizeof(*req));
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200184 }
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200185
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200186 spin_lock_irq(&dev->power.lock);
187 dev->power.constraints = NULL;
188 spin_unlock_irq(&dev->power.lock);
189
190 kfree(c->notifiers);
191 kfree(c);
192
193 out:
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200194 mutex_unlock(&dev_pm_qos_mtx);
195}
196
197/**
198 * dev_pm_qos_add_request - inserts new qos request into the list
199 * @dev: target device for the constraint
200 * @req: pointer to a preallocated handle
201 * @value: defines the qos request
202 *
203 * This function inserts a new entry in the device constraints list of
204 * requested qos performance characteristics. It recomputes the aggregate
205 * QoS expectations of parameters and initializes the dev_pm_qos_request
206 * handle. Caller needs to save this handle for later use in updates and
207 * removal.
208 *
209 * Returns 1 if the aggregated constraint value has changed,
210 * 0 if the aggregated constraint value has not changed,
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200211 * -EINVAL in case of wrong parameters, -ENOMEM if there's not enough memory
212 * to allocate for data structures, -ENODEV if the device has just been removed
213 * from the system.
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200214 */
215int dev_pm_qos_add_request(struct device *dev, struct dev_pm_qos_request *req,
Jean Pihetb66213c2011-08-25 15:35:47 +0200216 s32 value)
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200217{
218 int ret = 0;
219
220 if (!dev || !req) /*guard against callers passing in null */
221 return -EINVAL;
222
Guennadi Liakhovetskiaf4c7202011-11-10 00:44:18 +0100223 if (WARN(dev_pm_qos_request_active(req),
224 "%s() called for already added request\n", __func__))
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200225 return -EINVAL;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200226
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200227 req->dev = dev;
228
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200229 mutex_lock(&dev_pm_qos_mtx);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200230
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200231 if (!dev->power.constraints) {
232 if (dev->power.power_state.event == PM_EVENT_INVALID) {
233 /* The device has been removed from the system. */
234 req->dev = NULL;
235 ret = -ENODEV;
236 goto out;
237 } else {
238 /*
239 * Allocate the constraints data on the first call to
240 * add_request, i.e. only if the data is not already
241 * allocated and if the device has not been removed.
242 */
243 ret = dev_pm_qos_constraints_allocate(dev);
244 }
245 }
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200246
247 if (!ret)
Jean Pihetb66213c2011-08-25 15:35:47 +0200248 ret = apply_constraint(req, PM_QOS_ADD_REQ, value);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200249
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200250 out:
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200251 mutex_unlock(&dev_pm_qos_mtx);
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200252
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200253 return ret;
254}
255EXPORT_SYMBOL_GPL(dev_pm_qos_add_request);
256
257/**
258 * dev_pm_qos_update_request - modifies an existing qos request
259 * @req : handle to list element holding a dev_pm_qos request to use
260 * @new_value: defines the qos request
261 *
262 * Updates an existing dev PM qos request along with updating the
263 * target value.
264 *
265 * Attempts are made to make this code callable on hot code paths.
266 *
267 * Returns 1 if the aggregated constraint value has changed,
268 * 0 if the aggregated constraint value has not changed,
269 * -EINVAL in case of wrong parameters, -ENODEV if the device has been
270 * removed from the system
271 */
272int dev_pm_qos_update_request(struct dev_pm_qos_request *req,
273 s32 new_value)
274{
275 int ret = 0;
276
277 if (!req) /*guard against callers passing in null */
278 return -EINVAL;
279
Guennadi Liakhovetskiaf4c7202011-11-10 00:44:18 +0100280 if (WARN(!dev_pm_qos_request_active(req),
281 "%s() called for unknown object\n", __func__))
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200282 return -EINVAL;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200283
284 mutex_lock(&dev_pm_qos_mtx);
285
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200286 if (req->dev->power.constraints) {
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200287 if (new_value != req->node.prio)
Jean Pihetb66213c2011-08-25 15:35:47 +0200288 ret = apply_constraint(req, PM_QOS_UPDATE_REQ,
289 new_value);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200290 } else {
291 /* Return if the device has been removed */
292 ret = -ENODEV;
293 }
294
295 mutex_unlock(&dev_pm_qos_mtx);
296 return ret;
297}
298EXPORT_SYMBOL_GPL(dev_pm_qos_update_request);
299
300/**
301 * dev_pm_qos_remove_request - modifies an existing qos request
302 * @req: handle to request list element
303 *
304 * Will remove pm qos request from the list of constraints and
305 * recompute the current target value. Call this on slow code paths.
306 *
307 * Returns 1 if the aggregated constraint value has changed,
308 * 0 if the aggregated constraint value has not changed,
309 * -EINVAL in case of wrong parameters, -ENODEV if the device has been
310 * removed from the system
311 */
312int dev_pm_qos_remove_request(struct dev_pm_qos_request *req)
313{
314 int ret = 0;
315
316 if (!req) /*guard against callers passing in null */
317 return -EINVAL;
318
Guennadi Liakhovetskiaf4c7202011-11-10 00:44:18 +0100319 if (WARN(!dev_pm_qos_request_active(req),
320 "%s() called for unknown object\n", __func__))
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200321 return -EINVAL;
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200322
323 mutex_lock(&dev_pm_qos_mtx);
324
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200325 if (req->dev->power.constraints) {
Jean Pihetb66213c2011-08-25 15:35:47 +0200326 ret = apply_constraint(req, PM_QOS_REMOVE_REQ,
327 PM_QOS_DEFAULT_VALUE);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200328 memset(req, 0, sizeof(*req));
329 } else {
330 /* Return if the device has been removed */
331 ret = -ENODEV;
332 }
333
334 mutex_unlock(&dev_pm_qos_mtx);
335 return ret;
336}
337EXPORT_SYMBOL_GPL(dev_pm_qos_remove_request);
338
339/**
340 * dev_pm_qos_add_notifier - sets notification entry for changes to target value
341 * of per-device PM QoS constraints
342 *
343 * @dev: target device for the constraint
344 * @notifier: notifier block managed by caller.
345 *
346 * Will register the notifier into a notification chain that gets called
347 * upon changes to the target value for the device.
348 */
349int dev_pm_qos_add_notifier(struct device *dev, struct notifier_block *notifier)
350{
351 int retval = 0;
352
353 mutex_lock(&dev_pm_qos_mtx);
354
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200355 /* Silently return if the constraints object is not present. */
356 if (dev->power.constraints)
357 retval = blocking_notifier_chain_register(
358 dev->power.constraints->notifiers,
359 notifier);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200360
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200361 mutex_unlock(&dev_pm_qos_mtx);
362 return retval;
363}
364EXPORT_SYMBOL_GPL(dev_pm_qos_add_notifier);
365
366/**
367 * dev_pm_qos_remove_notifier - deletes notification for changes to target value
368 * of per-device PM QoS constraints
369 *
370 * @dev: target device for the constraint
371 * @notifier: notifier block to be removed.
372 *
373 * Will remove the notifier from the notification chain that gets called
374 * upon changes to the target value.
375 */
376int dev_pm_qos_remove_notifier(struct device *dev,
377 struct notifier_block *notifier)
378{
379 int retval = 0;
380
381 mutex_lock(&dev_pm_qos_mtx);
382
Rafael J. Wysocki1a9a9152011-09-29 22:29:44 +0200383 /* Silently return if the constraints object is not present. */
384 if (dev->power.constraints)
385 retval = blocking_notifier_chain_unregister(
386 dev->power.constraints->notifiers,
387 notifier);
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200388
Jean Pihet91ff4cb2011-08-25 15:35:41 +0200389 mutex_unlock(&dev_pm_qos_mtx);
390 return retval;
391}
392EXPORT_SYMBOL_GPL(dev_pm_qos_remove_notifier);
Jean Pihetb66213c2011-08-25 15:35:47 +0200393
394/**
395 * dev_pm_qos_add_global_notifier - sets notification entry for changes to
396 * target value of the PM QoS constraints for any device
397 *
398 * @notifier: notifier block managed by caller.
399 *
400 * Will register the notifier into a notification chain that gets called
401 * upon changes to the target value for any device.
402 */
403int dev_pm_qos_add_global_notifier(struct notifier_block *notifier)
404{
405 return blocking_notifier_chain_register(&dev_pm_notifiers, notifier);
406}
407EXPORT_SYMBOL_GPL(dev_pm_qos_add_global_notifier);
408
409/**
410 * dev_pm_qos_remove_global_notifier - deletes notification for changes to
411 * target value of PM QoS constraints for any device
412 *
413 * @notifier: notifier block to be removed.
414 *
415 * Will remove the notifier from the notification chain that gets called
416 * upon changes to the target value for any device.
417 */
418int dev_pm_qos_remove_global_notifier(struct notifier_block *notifier)
419{
420 return blocking_notifier_chain_unregister(&dev_pm_notifiers, notifier);
421}
422EXPORT_SYMBOL_GPL(dev_pm_qos_remove_global_notifier);
Rafael J. Wysocki40a5f8b2011-12-23 01:23:52 +0100423
424/**
425 * dev_pm_qos_add_ancestor_request - Add PM QoS request for device's ancestor.
426 * @dev: Device whose ancestor to add the request for.
427 * @req: Pointer to the preallocated handle.
428 * @value: Constraint latency value.
429 */
430int dev_pm_qos_add_ancestor_request(struct device *dev,
431 struct dev_pm_qos_request *req, s32 value)
432{
433 struct device *ancestor = dev->parent;
434 int error = -ENODEV;
435
436 while (ancestor && !ancestor->power.ignore_children)
437 ancestor = ancestor->parent;
438
439 if (ancestor)
440 error = dev_pm_qos_add_request(ancestor, req, value);
441
442 if (error)
443 req->dev = NULL;
444
445 return error;
446}
447EXPORT_SYMBOL_GPL(dev_pm_qos_add_ancestor_request);