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 34

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;
};

/**
35 36
 * pm_clk_add - Start using a device clock for power management.
 * @dev: Device whose clock is going to be used for power management.
37 38 39
 * @con_id: Connection ID of the clock.
 *
 * Add the clock represented by @con_id to the list of clocks used for
40
 * the power management of @dev.
41
 */
42
int pm_clk_add(struct device *dev, const char *con_id)
43
{
44
	struct pm_subsys_data *psd = dev_to_psd(dev);
45 46
	struct pm_clock_entry *ce;

47
	if (!psd)
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
		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;
		}
	}

66 67 68
	spin_lock_irq(&psd->lock);
	list_add_tail(&ce->node, &psd->clock_list);
	spin_unlock_irq(&psd->lock);
69 70 71 72
	return 0;
}

/**
73 74
 * __pm_clk_remove - Destroy PM clock entry.
 * @ce: PM clock entry to destroy.
75
 *
76 77
 * This routine must be called under the spinlock protecting the PM list of
 * clocks corresponding the the @ce's device.
78
 */
79
static void __pm_clk_remove(struct pm_clock_entry *ce)
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
{
	if (!ce)
		return;

	list_del(&ce->node);

	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);
	}

	if (ce->con_id)
		kfree(ce->con_id);

	kfree(ce);
}

/**
101 102
 * pm_clk_remove - Stop using a device clock for power management.
 * @dev: Device whose clock should not be used for PM any more.
103 104 105
 * @con_id: Connection ID of the clock.
 *
 * Remove the clock represented by @con_id from the list of clocks used for
106
 * the power management of @dev.
107
 */
108
void pm_clk_remove(struct device *dev, const char *con_id)
109
{
110
	struct pm_subsys_data *psd = dev_to_psd(dev);
111 112
	struct pm_clock_entry *ce;

113
	if (!psd)
114 115
		return;

116
	spin_lock_irq(&psd->lock);
117

118
	list_for_each_entry(ce, &psd->clock_list, node) {
119
		if (!con_id && !ce->con_id) {
120
			__pm_clk_remove(ce);
121 122 123 124
			break;
		} else if (!con_id || !ce->con_id) {
			continue;
		} else if (!strcmp(con_id, ce->con_id)) {
125
			__pm_clk_remove(ce);
126 127 128 129
			break;
		}
	}

130
	spin_unlock_irq(&psd->lock);
131 132 133
}

/**
134 135
 * pm_clk_init - Initialize a device's list of power management clocks.
 * @dev: Device to initialize the list of PM clocks for.
136
 *
137 138
 * Initialize the lock and clock_list members of the device's pm_subsys_data
 * object.
139
 */
140
void pm_clk_init(struct device *dev)
141
{
142
	struct pm_subsys_data *psd = dev_to_psd(dev);
143 144
	if (psd)
		INIT_LIST_HEAD(&psd->clock_list);
145 146 147 148 149 150 151 152 153 154 155
}

/**
 * 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)
{
156 157
	int ret = dev_pm_get_subsys_data(dev);
	return ret < 0 ? ret : 0;
158 159 160
}

/**
161 162
 * pm_clk_destroy - Destroy a device's list of power management clocks.
 * @dev: Device to destroy the list of PM clocks for.
163 164
 *
 * Clear the @dev's power.subsys_data field, remove the list of clock entries
165
 * from the struct pm_subsys_data object pointed to by it before and free
166 167
 * that object.
 */
168
void pm_clk_destroy(struct device *dev)
169
{
170
	struct pm_subsys_data *psd = dev_to_psd(dev);
171 172
	struct pm_clock_entry *ce, *c;

173
	if (!psd)
174 175
		return;

176
	spin_lock_irq(&psd->lock);
177

178
	list_for_each_entry_safe_reverse(ce, c, &psd->clock_list, node)
179
		__pm_clk_remove(ce);
180

181
	spin_unlock_irq(&psd->lock);
182

183
	dev_pm_put_subsys_data(dev);
184 185
}

186 187 188 189
#endif /* CONFIG_PM */

#ifdef CONFIG_PM_RUNTIME

190
/**
191
 * pm_clk_acquire - Acquire a device clock.
192 193 194
 * @dev: Device whose clock is to be acquired.
 * @con_id: Connection ID of the clock.
 */
195
static void pm_clk_acquire(struct device *dev,
196 197 198 199 200 201 202 203 204 205 206 207
				    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);
	}
}

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

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

219
	if (!psd)
220 221
		return 0;

222
	spin_lock_irqsave(&psd->lock, flags);
223

224
	list_for_each_entry_reverse(ce, &psd->clock_list, node) {
225
		if (ce->status == PCE_STATUS_NONE)
226
			pm_clk_acquire(dev, ce);
227 228 229 230 231 232 233

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

234
	spin_unlock_irqrestore(&psd->lock, flags);
235 236 237 238 239

	return 0;
}

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

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

251
	if (!psd)
252 253
		return 0;

254
	spin_lock_irqsave(&psd->lock, flags);
255

256
	list_for_each_entry(ce, &psd->clock_list, node) {
257
		if (ce->status == PCE_STATUS_NONE)
258
			pm_clk_acquire(dev, ce);
259 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);
}