clock_ops.c 11.1 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12
/*
 * drivers/base/power/clock_ops.c - Generic clock manipulation PM callbacks
 *
 * Copyright (c) 2011 Rafael J. Wysocki <rjw@sisk.pl>, Renesas Electronics Corp.
 *
 * This file is released under the GPLv2.
 */

#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/io.h>
#include <linux/pm.h>
13
#include <linux/pm_clock.h>
14 15 16 17
#include <linux/clk.h>
#include <linux/slab.h>
#include <linux/err.h>

18
#ifdef CONFIG_PM
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33

enum pce_status {
	PCE_STATUS_NONE = 0,
	PCE_STATUS_ACQUIRED,
	PCE_STATUS_ENABLED,
	PCE_STATUS_ERROR,
};

struct pm_clock_entry {
	struct list_head node;
	char *con_id;
	struct clk *clk;
	enum pce_status status;
};

34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
/**
 * pm_clk_acquire - Acquire a device clock.
 * @dev: Device whose clock is to be acquired.
 * @ce: PM clock entry corresponding to the clock.
 */
static void pm_clk_acquire(struct device *dev, struct pm_clock_entry *ce)
{
	ce->clk = clk_get(dev, ce->con_id);
	if (IS_ERR(ce->clk)) {
		ce->status = PCE_STATUS_ERROR;
	} else {
		ce->status = PCE_STATUS_ACQUIRED;
		dev_dbg(dev, "Clock %s managed by runtime PM.\n", ce->con_id);
	}
}

50
/**
51 52
 * pm_clk_add - Start using a device clock for power management.
 * @dev: Device whose clock is going to be used for power management.
53 54 55
 * @con_id: Connection ID of the clock.
 *
 * Add the clock represented by @con_id to the list of clocks used for
56
 * the power management of @dev.
57
 */
58
int pm_clk_add(struct device *dev, const char *con_id)
59
{
60
	struct pm_subsys_data *psd = dev_to_psd(dev);
61 62
	struct pm_clock_entry *ce;

63
	if (!psd)
64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
		return -EINVAL;

	ce = kzalloc(sizeof(*ce), GFP_KERNEL);
	if (!ce) {
		dev_err(dev, "Not enough memory for clock entry.\n");
		return -ENOMEM;
	}

	if (con_id) {
		ce->con_id = kstrdup(con_id, GFP_KERNEL);
		if (!ce->con_id) {
			dev_err(dev,
				"Not enough memory for clock connection ID.\n");
			kfree(ce);
			return -ENOMEM;
		}
	}

82 83
	pm_clk_acquire(dev, ce);

84 85 86
	spin_lock_irq(&psd->lock);
	list_add_tail(&ce->node, &psd->clock_list);
	spin_unlock_irq(&psd->lock);
87 88 89 90
	return 0;
}

/**
91 92
 * __pm_clk_remove - Destroy PM clock entry.
 * @ce: PM clock entry to destroy.
93
 */
94
static void __pm_clk_remove(struct pm_clock_entry *ce)
95 96 97 98 99 100 101 102 103 104 105 106
{
	if (!ce)
		return;

	if (ce->status < PCE_STATUS_ERROR) {
		if (ce->status == PCE_STATUS_ENABLED)
			clk_disable(ce->clk);

		if (ce->status >= PCE_STATUS_ACQUIRED)
			clk_put(ce->clk);
	}

107
	kfree(ce->con_id);
108 109 110 111
	kfree(ce);
}

/**
112 113
 * pm_clk_remove - Stop using a device clock for power management.
 * @dev: Device whose clock should not be used for PM any more.
114 115 116
 * @con_id: Connection ID of the clock.
 *
 * Remove the clock represented by @con_id from the list of clocks used for
117
 * the power management of @dev.
118
 */
119
void pm_clk_remove(struct device *dev, const char *con_id)
120
{
121
	struct pm_subsys_data *psd = dev_to_psd(dev);
122 123
	struct pm_clock_entry *ce;

124
	if (!psd)
125 126
		return;

127
	spin_lock_irq(&psd->lock);
128

129
	list_for_each_entry(ce, &psd->clock_list, node) {
130 131 132
		if (!con_id && !ce->con_id)
			goto remove;
		else if (!con_id || !ce->con_id)
133
			continue;
134 135
		else if (!strcmp(con_id, ce->con_id))
			goto remove;
136 137
	}

138
	spin_unlock_irq(&psd->lock);
139 140 141 142
	return;

 remove:
	list_del(&ce->node);
143
	spin_unlock_irq(&psd->lock);
144 145

	__pm_clk_remove(ce);
146 147 148
}

/**
149 150
 * pm_clk_init - Initialize a device's list of power management clocks.
 * @dev: Device to initialize the list of PM clocks for.
151
 *
152 153
 * Initialize the lock and clock_list members of the device's pm_subsys_data
 * object.
154
 */
155
void pm_clk_init(struct device *dev)
156
{
157
	struct pm_subsys_data *psd = dev_to_psd(dev);
158 159
	if (psd)
		INIT_LIST_HEAD(&psd->clock_list);
160 161 162 163 164 165 166 167 168 169 170
}

/**
 * pm_clk_create - Create and initialize a device's list of PM clocks.
 * @dev: Device to create and initialize the list of PM clocks for.
 *
 * Allocate a struct pm_subsys_data object, initialize its lock and clock_list
 * members and make the @dev's power.subsys_data field point to it.
 */
int pm_clk_create(struct device *dev)
{
171 172
	int ret = dev_pm_get_subsys_data(dev);
	return ret < 0 ? ret : 0;
173 174 175
}

/**
176 177
 * pm_clk_destroy - Destroy a device's list of power management clocks.
 * @dev: Device to destroy the list of PM clocks for.
178 179
 *
 * Clear the @dev's power.subsys_data field, remove the list of clock entries
180
 * from the struct pm_subsys_data object pointed to by it before and free
181 182
 * that object.
 */
183
void pm_clk_destroy(struct device *dev)
184
{
185
	struct pm_subsys_data *psd = dev_to_psd(dev);
186
	struct pm_clock_entry *ce, *c;
187
	struct list_head list;
188

189
	if (!psd)
190 191
		return;

192
	INIT_LIST_HEAD(&list);
193

194
	spin_lock_irq(&psd->lock);
195

196
	list_for_each_entry_safe_reverse(ce, c, &psd->clock_list, node)
197
		list_move(&ce->node, &list);
198

199
	spin_unlock_irq(&psd->lock);
200

201
	dev_pm_put_subsys_data(dev);
202 203 204 205 206

	list_for_each_entry_safe_reverse(ce, c, &list, node) {
		list_del(&ce->node);
		__pm_clk_remove(ce);
	}
207 208
}

209 210 211 212
#endif /* CONFIG_PM */

#ifdef CONFIG_PM_RUNTIME

213
/**
214
 * pm_clk_suspend - Disable clocks in a device's PM clock list.
215 216
 * @dev: Device to disable the clocks for.
 */
217
int pm_clk_suspend(struct device *dev)
218
{
219
	struct pm_subsys_data *psd = dev_to_psd(dev);
220
	struct pm_clock_entry *ce;
221
	unsigned long flags;
222 223 224

	dev_dbg(dev, "%s()\n", __func__);

225
	if (!psd)
226 227
		return 0;

228
	spin_lock_irqsave(&psd->lock, flags);
229

230
	list_for_each_entry_reverse(ce, &psd->clock_list, node) {
231 232 233 234 235 236
		if (ce->status < PCE_STATUS_ERROR) {
			clk_disable(ce->clk);
			ce->status = PCE_STATUS_ACQUIRED;
		}
	}

237
	spin_unlock_irqrestore(&psd->lock, flags);
238 239 240 241 242

	return 0;
}

/**
243
 * pm_clk_resume - Enable clocks in a device's PM clock list.
244 245
 * @dev: Device to enable the clocks for.
 */
246
int pm_clk_resume(struct device *dev)
247
{
248
	struct pm_subsys_data *psd = dev_to_psd(dev);
249
	struct pm_clock_entry *ce;
250
	unsigned long flags;
251 252 253

	dev_dbg(dev, "%s()\n", __func__);

254
	if (!psd)
255 256
		return 0;

257
	spin_lock_irqsave(&psd->lock, flags);
258

259
	list_for_each_entry(ce, &psd->clock_list, node) {
260 261 262 263 264 265
		if (ce->status < PCE_STATUS_ERROR) {
			clk_enable(ce->clk);
			ce->status = PCE_STATUS_ENABLED;
		}
	}

266
	spin_unlock_irqrestore(&psd->lock, flags);
267 268 269 270 271

	return 0;
}

/**
272
 * pm_clk_notify - Notify routine for device addition and removal.
273 274 275 276 277 278
 * @nb: Notifier block object this function is a member of.
 * @action: Operation being carried out by the caller.
 * @data: Device the routine is being run for.
 *
 * For this function to work, @nb must be a member of an object of type
 * struct pm_clk_notifier_block containing all of the requisite data.
279 280
 * Specifically, the pm_domain member of that object is copied to the device's
 * pm_domain field and its con_ids member is used to populate the device's list
281
 * of PM clocks, depending on @action.
282
 *
283
 * If the device's pm_domain field is already populated with a value different
284 285 286
 * from the one stored in the struct pm_clk_notifier_block object, the function
 * does nothing.
 */
287
static int pm_clk_notify(struct notifier_block *nb,
288 289 290 291
				 unsigned long action, void *data)
{
	struct pm_clk_notifier_block *clknb;
	struct device *dev = data;
292
	char **con_id;
293 294 295 296 297 298 299 300
	int error;

	dev_dbg(dev, "%s() %ld\n", __func__, action);

	clknb = container_of(nb, struct pm_clk_notifier_block, nb);

	switch (action) {
	case BUS_NOTIFY_ADD_DEVICE:
301
		if (dev->pm_domain)
302 303
			break;

304
		error = pm_clk_create(dev);
305 306 307
		if (error)
			break;

308
		dev->pm_domain = clknb->pm_domain;
309
		if (clknb->con_ids[0]) {
310
			for (con_id = clknb->con_ids; *con_id; con_id++)
311
				pm_clk_add(dev, *con_id);
312
		} else {
313
			pm_clk_add(dev, NULL);
314 315 316 317
		}

		break;
	case BUS_NOTIFY_DEL_DEVICE:
318
		if (dev->pm_domain != clknb->pm_domain)
319 320
			break;

321
		dev->pm_domain = NULL;
322
		pm_clk_destroy(dev);
323 324 325 326 327 328 329 330
		break;
	}

	return 0;
}

#else /* !CONFIG_PM_RUNTIME */

331 332 333
#ifdef CONFIG_PM

/**
334
 * pm_clk_suspend - Disable clocks in a device's PM clock list.
335 336
 * @dev: Device to disable the clocks for.
 */
337
int pm_clk_suspend(struct device *dev)
338
{
339
	struct pm_subsys_data *psd = dev_to_psd(dev);
340
	struct pm_clock_entry *ce;
341
	unsigned long flags;
342 343 344 345

	dev_dbg(dev, "%s()\n", __func__);

	/* If there is no driver, the clocks are already disabled. */
346
	if (!psd || !dev->driver)
347 348
		return 0;

349
	spin_lock_irqsave(&psd->lock, flags);
350

351
	list_for_each_entry_reverse(ce, &psd->clock_list, node)
352 353
		clk_disable(ce->clk);

354
	spin_unlock_irqrestore(&psd->lock, flags);
355 356 357 358 359

	return 0;
}

/**
360
 * pm_clk_resume - Enable clocks in a device's PM clock list.
361 362
 * @dev: Device to enable the clocks for.
 */
363
int pm_clk_resume(struct device *dev)
364
{
365
	struct pm_subsys_data *psd = dev_to_psd(dev);
366
	struct pm_clock_entry *ce;
367
	unsigned long flags;
368 369 370 371

	dev_dbg(dev, "%s()\n", __func__);

	/* If there is no driver, the clocks should remain disabled. */
372
	if (!psd || !dev->driver)
373 374
		return 0;

375
	spin_lock_irqsave(&psd->lock, flags);
376

377
	list_for_each_entry(ce, &psd->clock_list, node)
378 379
		clk_enable(ce->clk);

380
	spin_unlock_irqrestore(&psd->lock, flags);
381 382 383 384 385 386

	return 0;
}

#endif /* CONFIG_PM */

387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
/**
 * enable_clock - Enable a device clock.
 * @dev: Device whose clock is to be enabled.
 * @con_id: Connection ID of the clock.
 */
static void enable_clock(struct device *dev, const char *con_id)
{
	struct clk *clk;

	clk = clk_get(dev, con_id);
	if (!IS_ERR(clk)) {
		clk_enable(clk);
		clk_put(clk);
		dev_info(dev, "Runtime PM disabled, clock forced on.\n");
	}
}

/**
 * disable_clock - Disable a device clock.
 * @dev: Device whose clock is to be disabled.
 * @con_id: Connection ID of the clock.
 */
static void disable_clock(struct device *dev, const char *con_id)
{
	struct clk *clk;

	clk = clk_get(dev, con_id);
	if (!IS_ERR(clk)) {
		clk_disable(clk);
		clk_put(clk);
		dev_info(dev, "Runtime PM disabled, clock forced off.\n");
	}
}

/**
422
 * pm_clk_notify - Notify routine for device addition and removal.
423 424 425 426 427 428 429 430 431
 * @nb: Notifier block object this function is a member of.
 * @action: Operation being carried out by the caller.
 * @data: Device the routine is being run for.
 *
 * For this function to work, @nb must be a member of an object of type
 * struct pm_clk_notifier_block containing all of the requisite data.
 * Specifically, the con_ids member of that object is used to enable or disable
 * the device's clocks, depending on @action.
 */
432
static int pm_clk_notify(struct notifier_block *nb,
433 434 435 436
				 unsigned long action, void *data)
{
	struct pm_clk_notifier_block *clknb;
	struct device *dev = data;
437
	char **con_id;
438 439 440 441 442 443

	dev_dbg(dev, "%s() %ld\n", __func__, action);

	clknb = container_of(nb, struct pm_clk_notifier_block, nb);

	switch (action) {
444
	case BUS_NOTIFY_BIND_DRIVER:
445
		if (clknb->con_ids[0]) {
446 447
			for (con_id = clknb->con_ids; *con_id; con_id++)
				enable_clock(dev, *con_id);
448 449 450 451
		} else {
			enable_clock(dev, NULL);
		}
		break;
452
	case BUS_NOTIFY_UNBOUND_DRIVER:
453
		if (clknb->con_ids[0]) {
454 455
			for (con_id = clknb->con_ids; *con_id; con_id++)
				disable_clock(dev, *con_id);
456 457 458 459 460 461 462 463 464 465 466 467
		} else {
			disable_clock(dev, NULL);
		}
		break;
	}

	return 0;
}

#endif /* !CONFIG_PM_RUNTIME */

/**
468
 * pm_clk_add_notifier - Add bus type notifier for power management clocks.
469 470 471 472
 * @bus: Bus type to add the notifier to.
 * @clknb: Notifier to be added to the given bus type.
 *
 * The nb member of @clknb is not expected to be initialized and its
473
 * notifier_call member will be replaced with pm_clk_notify().  However,
474 475 476
 * the remaining members of @clknb should be populated prior to calling this
 * routine.
 */
477
void pm_clk_add_notifier(struct bus_type *bus,
478 479 480 481 482
				 struct pm_clk_notifier_block *clknb)
{
	if (!bus || !clknb)
		return;

483
	clknb->nb.notifier_call = pm_clk_notify;
484 485
	bus_register_notifier(bus, &clknb->nb);
}