| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * rmobile power management support | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2012  Renesas Solutions Corp. | 
					
						
							|  |  |  |  * Copyright (C) 2012  Kuninori Morimoto <kuninori.morimoto.gx@renesas.com> | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  |  * Copyright (C) 2014  Glider bvba | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * based on pm-sh7372.c | 
					
						
							|  |  |  |  *  Copyright (C) 2011 Magnus Damm | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is subject to the terms and conditions of the GNU General Public | 
					
						
							|  |  |  |  * License.  See the file "COPYING" in the main directory of this archive | 
					
						
							|  |  |  |  * for more details. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <linux/console.h>
 | 
					
						
							|  |  |  | #include <linux/delay.h>
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | #include <linux/of.h>
 | 
					
						
							|  |  |  | #include <linux/of_address.h>
 | 
					
						
							|  |  |  | #include <linux/of_platform.h>
 | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | #include <linux/platform_device.h>
 | 
					
						
							|  |  |  | #include <linux/pm.h>
 | 
					
						
							|  |  |  | #include <linux/pm_clock.h>
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | #include <asm/io.h>
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-17 16:47:45 +09:00
										 |  |  | #include "pm-rmobile.h"
 | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* SYSC */ | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | #define SPDCR		0x08	/* SYS Power Down Control Register */
 | 
					
						
							|  |  |  | #define SWUCR		0x14	/* SYS Wakeup Control Register */
 | 
					
						
							|  |  |  | #define PSTR		0x80	/* Power Status Register */
 | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define PSTR_RETRIES	100
 | 
					
						
							|  |  |  | #define PSTR_DELAY_US	10
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int rmobile_pd_power_down(struct generic_pm_domain *genpd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rmobile_pm_domain *rmobile_pd = to_rmobile_pd(genpd); | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	unsigned int mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (rmobile_pd->bit_shift == ~0) | 
					
						
							|  |  |  | 		return -EBUSY; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	mask = 1 << rmobile_pd->bit_shift; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 	if (rmobile_pd->suspend) { | 
					
						
							|  |  |  | 		int ret = rmobile_pd->suspend(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (ret) | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 	if (__raw_readl(rmobile_pd->base + PSTR) & mask) { | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 		unsigned int retry_count; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 		__raw_writel(mask, rmobile_pd->base + SPDCR); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		for (retry_count = PSTR_RETRIES; retry_count; retry_count--) { | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 			if (!(__raw_readl(rmobile_pd->base + SPDCR) & mask)) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 			cpu_relax(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!rmobile_pd->no_debug) | 
					
						
							|  |  |  | 		pr_debug("%s: Power off, 0x%08x -> PSTR = 0x%08x\n", | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 			 genpd->name, mask, | 
					
						
							|  |  |  | 			 __raw_readl(rmobile_pd->base + PSTR)); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __rmobile_pd_power_up(struct rmobile_pm_domain *rmobile_pd, | 
					
						
							|  |  |  | 				 bool do_resume) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	unsigned int mask; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 	unsigned int retry_count; | 
					
						
							|  |  |  | 	int ret = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	if (rmobile_pd->bit_shift == ~0) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mask = 1 << rmobile_pd->bit_shift; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 	if (__raw_readl(rmobile_pd->base + PSTR) & mask) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 	__raw_writel(mask, rmobile_pd->base + SWUCR); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (retry_count = 2 * PSTR_RETRIES; retry_count; retry_count--) { | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 		if (!(__raw_readl(rmobile_pd->base + SWUCR) & mask)) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		if (retry_count > PSTR_RETRIES) | 
					
						
							|  |  |  | 			udelay(PSTR_DELAY_US); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			cpu_relax(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!retry_count) | 
					
						
							|  |  |  | 		ret = -EIO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!rmobile_pd->no_debug) | 
					
						
							|  |  |  | 		pr_debug("%s: Power on, 0x%08x -> PSTR = 0x%08x\n", | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:44 +01:00
										 |  |  | 			 rmobile_pd->genpd.name, mask, | 
					
						
							|  |  |  | 			 __raw_readl(rmobile_pd->base + PSTR)); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  | 	if (ret == 0 && rmobile_pd->resume && do_resume) | 
					
						
							|  |  |  | 		rmobile_pd->resume(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int rmobile_pd_power_up(struct generic_pm_domain *genpd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return __rmobile_pd_power_up(to_rmobile_pd(genpd), true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool rmobile_pd_active_wakeup(struct device *dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-04-11 17:26:41 +02:00
										 |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:42 +01:00
										 |  |  | static int rmobile_pd_attach_dev(struct generic_pm_domain *domain, | 
					
						
							|  |  |  | 				 struct device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	error = pm_clk_create(dev); | 
					
						
							|  |  |  | 	if (error) { | 
					
						
							|  |  |  | 		dev_err(dev, "pm_clk_create failed %d\n", error); | 
					
						
							|  |  |  | 		return error; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	error = pm_clk_add(dev, NULL); | 
					
						
							|  |  |  | 	if (error) { | 
					
						
							|  |  |  | 		dev_err(dev, "pm_clk_add failed %d\n", error); | 
					
						
							|  |  |  | 		goto fail; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fail: | 
					
						
							|  |  |  | 	pm_clk_destroy(dev); | 
					
						
							|  |  |  | 	return error; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void rmobile_pd_detach_dev(struct generic_pm_domain *domain, | 
					
						
							|  |  |  | 				  struct device *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	pm_clk_destroy(dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-07 01:15:02 +02:00
										 |  |  | static void rmobile_init_pm_domain(struct rmobile_pm_domain *rmobile_pd) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct generic_pm_domain *genpd = &rmobile_pd->genpd; | 
					
						
							|  |  |  | 	struct dev_power_governor *gov = rmobile_pd->gov; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-01 12:50:23 +01:00
										 |  |  | 	genpd->flags = GENPD_FLAG_PM_CLK; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 	pm_genpd_init(genpd, gov ? : &simple_qos_governor, false); | 
					
						
							|  |  |  | 	genpd->dev_ops.active_wakeup	= rmobile_pd_active_wakeup; | 
					
						
							|  |  |  | 	genpd->power_off		= rmobile_pd_power_down; | 
					
						
							|  |  |  | 	genpd->power_on			= rmobile_pd_power_up; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:42 +01:00
										 |  |  | 	genpd->attach_dev		= rmobile_pd_attach_dev; | 
					
						
							|  |  |  | 	genpd->detach_dev		= rmobile_pd_detach_dev; | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | 	__rmobile_pd_power_up(rmobile_pd, false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | #ifdef CONFIG_ARCH_SHMOBILE_LEGACY
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-07 01:10:22 +02:00
										 |  |  | void rmobile_init_domains(struct rmobile_pm_domain domains[], int num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (j = 0; j < num; j++) | 
					
						
							|  |  |  | 		rmobile_init_pm_domain(&domains[j]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-08 00:27:52 +02:00
										 |  |  | void rmobile_add_device_to_domain_td(const char *domain_name, | 
					
						
							|  |  |  | 				     struct platform_device *pdev, | 
					
						
							|  |  |  | 				     struct gpd_timing_data *td) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct device *dev = &pdev->dev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-08 00:27:52 +02:00
										 |  |  | 	__pm_genpd_name_add_device(domain_name, dev, td); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-15 20:56:26 +02:00
										 |  |  | void rmobile_add_devices_to_domains(struct pm_domain_device data[], | 
					
						
							|  |  |  | 				    int size) | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-08-15 20:56:26 +02:00
										 |  |  | 	struct gpd_timing_data latencies = { | 
					
						
							|  |  |  | 		.stop_latency_ns = DEFAULT_DEV_LATENCY_NS, | 
					
						
							|  |  |  | 		.start_latency_ns = DEFAULT_DEV_LATENCY_NS, | 
					
						
							|  |  |  | 		.save_state_latency_ns = DEFAULT_DEV_LATENCY_NS, | 
					
						
							|  |  |  | 		.restore_state_latency_ns = DEFAULT_DEV_LATENCY_NS, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	int j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (j = 0; j < size; j++) | 
					
						
							|  |  |  | 		rmobile_add_device_to_domain_td(data[j].domain_name, | 
					
						
							|  |  |  | 						data[j].pdev, &latencies); | 
					
						
							| 
									
										
										
										
											2012-07-05 01:24:46 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #else /* !CONFIG_ARCH_SHMOBILE_LEGACY */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | static int rmobile_pd_suspend_busy(void) | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | 	 * This domain should not be turned off. | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	 */ | 
					
						
							|  |  |  | 	return -EBUSY; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int rmobile_pd_suspend_console(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Serial consoles make use of SCIF hardware located in this domain, | 
					
						
							|  |  |  | 	 * hence keep the power domain on if "no_console_suspend" is set. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	return console_suspend_enabled ? 0 : -EBUSY; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | enum pd_types { | 
					
						
							|  |  |  | 	PD_NORMAL, | 
					
						
							|  |  |  | 	PD_CPU, | 
					
						
							|  |  |  | 	PD_CONSOLE, | 
					
						
							|  |  |  | 	PD_DEBUG, | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:22 +01:00
										 |  |  | 	PD_MEMCTL, | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | #define MAX_NUM_SPECIAL_PDS	16
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | static struct special_pd { | 
					
						
							|  |  |  | 	struct device_node *pd; | 
					
						
							|  |  |  | 	enum pd_types type; | 
					
						
							|  |  |  | } special_pds[MAX_NUM_SPECIAL_PDS] __initdata; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | static unsigned int num_special_pds __initdata; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:22 +01:00
										 |  |  | static const struct of_device_id special_ids[] __initconst = { | 
					
						
							|  |  |  | 	{ .compatible = "arm,coresight-etm3x", .data = (void *)PD_DEBUG }, | 
					
						
							|  |  |  | 	{ .compatible = "renesas,dbsc-r8a73a4", .data = (void *)PD_MEMCTL, }, | 
					
						
							|  |  |  | 	{ .compatible = "renesas,dbsc3-r8a7740", .data = (void *)PD_MEMCTL, }, | 
					
						
							|  |  |  | 	{ .compatible = "renesas,sbsc-sh73a0", .data = (void *)PD_MEMCTL, }, | 
					
						
							|  |  |  | 	{ /* sentinel */ }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | static void __init add_special_pd(struct device_node *np, enum pd_types type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 	struct device_node *pd; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	pd = of_parse_phandle(np, "power-domains", 0); | 
					
						
							|  |  |  | 	if (!pd) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	for (i = 0; i < num_special_pds; i++) | 
					
						
							|  |  |  | 		if (pd == special_pds[i].pd && type == special_pds[i].type) { | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 			of_node_put(pd); | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 			return; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	if (num_special_pds == ARRAY_SIZE(special_pds)) { | 
					
						
							|  |  |  | 		pr_warn("Too many special PM domains\n"); | 
					
						
							|  |  |  | 		of_node_put(pd); | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	pr_debug("Special PM domain %s type %d for %s\n", pd->name, type, | 
					
						
							|  |  |  | 		 np->full_name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	special_pds[num_special_pds].pd = pd; | 
					
						
							|  |  |  | 	special_pds[num_special_pds].type = type; | 
					
						
							|  |  |  | 	num_special_pds++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __init get_special_pds(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device_node *np; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:22 +01:00
										 |  |  | 	const struct of_device_id *id; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* PM domains containing CPUs */ | 
					
						
							|  |  |  | 	for_each_node_by_type(np, "cpu") | 
					
						
							|  |  |  | 		add_special_pd(np, PD_CPU); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	/* PM domain containing console */ | 
					
						
							|  |  |  | 	if (of_stdout) | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 		add_special_pd(of_stdout, PD_CONSOLE); | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:22 +01:00
										 |  |  | 	/* PM domains containing other special devices */ | 
					
						
							|  |  |  | 	for_each_matching_node_and_match(np, special_ids, &id) | 
					
						
							|  |  |  | 		add_special_pd(np, (enum pd_types)id->data); | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __init put_special_pds(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	for (i = 0; i < num_special_pds; i++) | 
					
						
							|  |  |  | 		of_node_put(special_pds[i].pd); | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | static enum pd_types __init pd_type(const struct device_node *pd) | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	for (i = 0; i < num_special_pds; i++) | 
					
						
							|  |  |  | 		if (pd == special_pds[i].pd) | 
					
						
							|  |  |  | 			return special_pds[i].type; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	return PD_NORMAL; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __init rmobile_setup_pm_domain(struct device_node *np, | 
					
						
							|  |  |  | 					   struct rmobile_pm_domain *pd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const char *name = pd->genpd.name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	switch (pd_type(np)) { | 
					
						
							|  |  |  | 	case PD_CPU: | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * This domain contains the CPU core and therefore it should | 
					
						
							|  |  |  | 		 * only be turned off if the CPU is not in use. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 		pr_debug("PM domain %s contains CPU\n", name); | 
					
						
							|  |  |  | 		pd->gov = &pm_domain_always_on_gov; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | 		pd->suspend = rmobile_pd_suspend_busy; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case PD_CONSOLE: | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 		pr_debug("PM domain %s contains serial console\n", name); | 
					
						
							|  |  |  | 		pd->gov = &pm_domain_always_on_gov; | 
					
						
							|  |  |  | 		pd->suspend = rmobile_pd_suspend_console; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case PD_DEBUG: | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * This domain contains the Coresight-ETM hardware block and | 
					
						
							|  |  |  | 		 * therefore it should only be turned off if the debug module | 
					
						
							|  |  |  | 		 * is not in use. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 		pr_debug("PM domain %s contains Coresight-ETM\n", name); | 
					
						
							|  |  |  | 		pd->gov = &pm_domain_always_on_gov; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:20 +01:00
										 |  |  | 		pd->suspend = rmobile_pd_suspend_busy; | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:22 +01:00
										 |  |  | 	case PD_MEMCTL: | 
					
						
							|  |  |  | 		/*
 | 
					
						
							|  |  |  | 		 * This domain contains a memory-controller and therefore it | 
					
						
							|  |  |  | 		 * should only be turned off if memory is not in use. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		pr_debug("PM domain %s contains MEMCTL\n", name); | 
					
						
							|  |  |  | 		pd->gov = &pm_domain_always_on_gov; | 
					
						
							|  |  |  | 		pd->suspend = rmobile_pd_suspend_busy; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-14 13:11:21 +01:00
										 |  |  | 	case PD_NORMAL: | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											2014-12-03 14:41:45 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rmobile_init_pm_domain(pd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init rmobile_add_pm_domains(void __iomem *base, | 
					
						
							|  |  |  | 					 struct device_node *parent, | 
					
						
							|  |  |  | 					 struct generic_pm_domain *genpd_parent) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device_node *np; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for_each_child_of_node(parent, np) { | 
					
						
							|  |  |  | 		struct rmobile_pm_domain *pd; | 
					
						
							|  |  |  | 		u32 idx = ~0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (of_property_read_u32(np, "reg", &idx)) { | 
					
						
							|  |  |  | 			/* always-on domain */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		pd = kzalloc(sizeof(*pd), GFP_KERNEL); | 
					
						
							|  |  |  | 		if (!pd) | 
					
						
							|  |  |  | 			return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		pd->genpd.name = np->name; | 
					
						
							|  |  |  | 		pd->base = base; | 
					
						
							|  |  |  | 		pd->bit_shift = idx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		rmobile_setup_pm_domain(np, pd); | 
					
						
							|  |  |  | 		if (genpd_parent) | 
					
						
							|  |  |  | 			pm_genpd_add_subdomain(genpd_parent, &pd->genpd); | 
					
						
							|  |  |  | 		of_genpd_add_provider_simple(np, &pd->genpd); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		rmobile_add_pm_domains(base, np, &pd->genpd); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init rmobile_init_pm_domains(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device_node *np, *pmd; | 
					
						
							|  |  |  | 	bool scanned = false; | 
					
						
							|  |  |  | 	void __iomem *base; | 
					
						
							|  |  |  | 	int ret = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for_each_compatible_node(np, NULL, "renesas,sysc-rmobile") { | 
					
						
							|  |  |  | 		base = of_iomap(np, 0); | 
					
						
							|  |  |  | 		if (!base) { | 
					
						
							|  |  |  | 			pr_warn("%s cannot map reg 0\n", np->full_name); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		pmd = of_get_child_by_name(np, "pm-domains"); | 
					
						
							|  |  |  | 		if (!pmd) { | 
					
						
							|  |  |  | 			pr_warn("%s lacks pm-domains node\n", np->full_name); | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!scanned) { | 
					
						
							|  |  |  | 			/* Find PM domains containing special blocks */ | 
					
						
							|  |  |  | 			get_special_pds(); | 
					
						
							|  |  |  | 			scanned = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ret = rmobile_add_pm_domains(base, pmd, NULL); | 
					
						
							|  |  |  | 		of_node_put(pmd); | 
					
						
							|  |  |  | 		if (ret) { | 
					
						
							|  |  |  | 			of_node_put(np); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	put_special_pds(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | core_initcall(rmobile_init_pm_domains); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* !CONFIG_ARCH_SHMOBILE_LEGACY */
 |