pm.c 5.4 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * pm.c - Common OMAP2+ power management-related code
 *
 * Copyright (C) 2010 Texas Instruments, Inc.
 * Copyright (C) 2010 Nokia Corporation
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */

#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/io.h>
#include <linux/err.h>
16
#include <linux/opp.h>
17
#include <linux/export.h>
18 19 20 21 22

#include <plat/omap-pm.h>
#include <plat/omap_device.h>
#include <plat/common.h>

23
#include "voltage.h"
24
#include "powerdomain.h"
25
#include "clockdomain.h"
26
#include "pm.h"
27

28 29
static struct omap_device_pm_latency *pm_lats;

30
static int _init_omap_device(char *name)
31 32
{
	struct omap_hwmod *oh;
33
	struct platform_device *pdev;
34 35 36 37 38 39

	oh = omap_hwmod_lookup(name);
	if (WARN(!oh, "%s: could not find omap_hwmod for %s\n",
		 __func__, name))
		return -ENODEV;

40 41
	pdev = omap_device_build(oh->name, 0, oh, NULL, 0, pm_lats, 0, false);
	if (WARN(IS_ERR(pdev), "%s: could not build omap_device for %s\n",
42 43 44 45 46 47 48 49 50 51 52
		 __func__, name))
		return -ENODEV;

	return 0;
}

/*
 * Build omap_devices for processors and bus.
 */
static void omap2_init_processor_devices(void)
{
53
	_init_omap_device("mpu");
54
	if (omap3_has_iva())
55
		_init_omap_device("iva");
56

57
	if (cpu_is_omap44xx()) {
58 59 60
		_init_omap_device("l3_main_1");
		_init_omap_device("dsp");
		_init_omap_device("iva");
61
	} else {
62
		_init_omap_device("l3_main");
63
	}
64 65
}

66 67 68 69
/* Types of sleep_switch used in omap_set_pwrdm_state */
#define FORCEWAKEUP_SWITCH	0
#define LOWPOWERSTATE_SWITCH	1

70 71
/*
 * This sets pwrdm state (other than mpu & core. Currently only ON &
72
 * RET are supported.
73 74 75 76
 */
int omap_set_pwrdm_state(struct powerdomain *pwrdm, u32 state)
{
	u32 cur_state;
77
	int sleep_switch = -1;
78
	int ret = 0;
79
	int hwsup = 0;
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94

	if (pwrdm == NULL || IS_ERR(pwrdm))
		return -EINVAL;

	while (!(pwrdm->pwrsts & (1 << state))) {
		if (state == PWRDM_POWER_OFF)
			return ret;
		state--;
	}

	cur_state = pwrdm_read_next_pwrst(pwrdm);
	if (cur_state == state)
		return ret;

	if (pwrdm_read_pwrst(pwrdm) < PWRDM_POWER_ON) {
95 96 97 98
		if ((pwrdm_read_pwrst(pwrdm) > state) &&
			(pwrdm->flags & PWRDM_HAS_LOWPOWERSTATECHANGE)) {
			sleep_switch = LOWPOWERSTATE_SWITCH;
		} else {
99
			hwsup = clkdm_in_hwsup(pwrdm->pwrdm_clkdms[0]);
100
			clkdm_wakeup(pwrdm->pwrdm_clkdms[0]);
101 102
			sleep_switch = FORCEWAKEUP_SWITCH;
		}
103 104 105 106
	}

	ret = pwrdm_set_next_pwrst(pwrdm, state);
	if (ret) {
107 108
		pr_err("%s: unable to set state of powerdomain: %s\n",
		       __func__, pwrdm->name);
109 110 111
		goto err;
	}

112 113
	switch (sleep_switch) {
	case FORCEWAKEUP_SWITCH:
114
		if (hwsup)
115
			clkdm_allow_idle(pwrdm->pwrdm_clkdms[0]);
116
		else
117
			clkdm_sleep(pwrdm->pwrdm_clkdms[0]);
118 119 120 121 122 123
		break;
	case LOWPOWERSTATE_SWITCH:
		pwrdm_set_lowpwrstchange(pwrdm);
		break;
	default:
		return ret;
124 125
	}

126
	pwrdm_state_switch(pwrdm);
127 128 129 130
err:
	return ret;
}

131
/*
132
 * This API is to be called during init to set the various voltage
133 134 135
 * domains to the voltage as per the opp table. Typically we boot up
 * at the nominal voltage. So this function finds out the rate of
 * the clock associated with the voltage domain, finds out the correct
136
 * opp entry and sets the voltage domain to the voltage specified
137 138 139
 * in the opp entry
 */
static int __init omap2_set_init_voltage(char *vdd_name, char *clk_name,
140
					 const char *oh_name)
141 142 143 144 145
{
	struct voltagedomain *voltdm;
	struct clk *clk;
	struct opp *opp;
	unsigned long freq, bootup_volt;
146
	struct device *dev;
147

148
	if (!vdd_name || !clk_name || !oh_name) {
149
		pr_err("%s: invalid parameters\n", __func__);
150 151 152
		goto exit;
	}

153 154 155 156 157 158 159
	dev = omap_device_get_by_hwmod_name(oh_name);
	if (IS_ERR(dev)) {
		pr_err("%s: Unable to get dev pointer for hwmod %s\n",
			__func__, oh_name);
		goto exit;
	}

160
	voltdm = voltdm_lookup(vdd_name);
161
	if (IS_ERR(voltdm)) {
162
		pr_err("%s: unable to get vdd pointer for vdd_%s\n",
163 164 165 166 167 168
			__func__, vdd_name);
		goto exit;
	}

	clk =  clk_get(NULL, clk_name);
	if (IS_ERR(clk)) {
169
		pr_err("%s: unable to get clk %s\n", __func__, clk_name);
170 171 172 173 174 175 176 177
		goto exit;
	}

	freq = clk->rate;
	clk_put(clk);

	opp = opp_find_freq_ceil(dev, &freq);
	if (IS_ERR(opp)) {
178
		pr_err("%s: unable to find boot up OPP for vdd_%s\n",
179 180 181 182 183 184
			__func__, vdd_name);
		goto exit;
	}

	bootup_volt = opp_get_voltage(opp);
	if (!bootup_volt) {
185
		pr_err("%s: unable to find voltage corresponding "
186 187 188 189
			"to the bootup OPP for vdd_%s\n", __func__, vdd_name);
		goto exit;
	}

190
	voltdm_scale(voltdm, bootup_volt);
191 192 193
	return 0;

exit:
194
	pr_err("%s: unable to set vdd_%s\n", __func__, vdd_name);
195 196 197 198 199 200 201 202
	return -EINVAL;
}

static void __init omap3_init_voltages(void)
{
	if (!cpu_is_omap34xx())
		return;

203 204
	omap2_set_init_voltage("mpu_iva", "dpll1_ck", "mpu");
	omap2_set_init_voltage("core", "l3_ick", "l3_main");
205 206
}

207 208 209 210 211
static void __init omap4_init_voltages(void)
{
	if (!cpu_is_omap44xx())
		return;

212 213 214
	omap2_set_init_voltage("mpu", "dpll_mpu_ck", "mpu");
	omap2_set_init_voltage("core", "l3_div_ck", "l3_main_1");
	omap2_set_init_voltage("iva", "dpll_iva_m5x2_ck", "iva");
215 216
}

217 218
static int __init omap2_common_pm_init(void)
{
219 220
	if (!of_have_populated_dt())
		omap2_init_processor_devices();
221 222 223 224
	omap_pm_if_init();

	return 0;
}
225
postcore_initcall(omap2_common_pm_init);
226

227 228
static int __init omap2_common_pm_late_init(void)
{
229 230
	/* Init the OMAP TWL parameters */
	omap3_twl_init();
231
	omap4_twl_init();
232

233
	/* Init the voltage layer */
234
	omap_voltage_late_init();
235 236 237

	/* Initialize the voltages */
	omap3_init_voltages();
238
	omap4_init_voltages();
239

240
	/* Smartreflex device init */
241
	omap_devinit_smartreflex();
242 243 244 245

	return 0;
}
late_initcall(omap2_common_pm_late_init);