diff options
Diffstat (limited to 'arch/arm/mach-at91/setup.c')
| -rw-r--r-- | arch/arm/mach-at91/setup.c | 159 | 
1 files changed, 159 insertions, 0 deletions
diff --git a/arch/arm/mach-at91/setup.c b/arch/arm/mach-at91/setup.c index 372396c2ecb..97cc04dc807 100644 --- a/arch/arm/mach-at91/setup.c +++ b/arch/arm/mach-at91/setup.c @@ -9,7 +9,9 @@  #include <linux/io.h>  #include <linux/mm.h>  #include <linux/pm.h> +#include <linux/of_address.h> +#include <asm/system_misc.h>  #include <asm/mach/map.h>  #include <mach/hardware.h> @@ -51,6 +53,19 @@ void __init at91_init_interrupts(unsigned int *priority)  	at91_gpio_irq_setup();  } +void __iomem *at91_ramc_base[2]; + +void __init at91_ioremap_ramc(int id, u32 addr, u32 size) +{ +	if (id < 0 || id > 1) { +		pr_emerg("Wrong RAM controller id (%d), cannot continue\n", id); +		BUG(); +	} +	at91_ramc_base[id] = ioremap(addr, size); +	if (!at91_ramc_base[id]) +		panic("Impossible to ioremap ramc.%d 0x%x\n", id, addr); +} +  static struct map_desc sram_desc[2] __initdata;  void __init at91_init_sram(int bank, unsigned long base, unsigned int length) @@ -285,6 +300,150 @@ void __init at91_ioremap_matrix(u32 base_addr)  		panic("Impossible to ioremap at91_matrix_base\n");  } +#if defined(CONFIG_OF) +static struct of_device_id rstc_ids[] = { +	{ .compatible = "atmel,at91sam9260-rstc", .data = at91sam9_alt_restart }, +	{ .compatible = "atmel,at91sam9g45-rstc", .data = at91sam9g45_restart }, +	{ /*sentinel*/ } +}; + +static void at91_dt_rstc(void) +{ +	struct device_node *np; +	const struct of_device_id *of_id; + +	np = of_find_matching_node(NULL, rstc_ids); +	if (!np) +		panic("unable to find compatible rstc node in dtb\n"); + +	at91_rstc_base = of_iomap(np, 0); +	if (!at91_rstc_base) +		panic("unable to map rstc cpu registers\n"); + +	of_id = of_match_node(rstc_ids, np); +	if (!of_id) +		panic("AT91: rtsc no restart function availlable\n"); + +	arm_pm_restart = of_id->data; + +	of_node_put(np); +} + +static struct of_device_id ramc_ids[] = { +	{ .compatible = "atmel,at91sam9260-sdramc" }, +	{ .compatible = "atmel,at91sam9g45-ddramc" }, +	{ /*sentinel*/ } +}; + +static void at91_dt_ramc(void) +{ +	struct device_node *np; + +	np = of_find_matching_node(NULL, ramc_ids); +	if (!np) +		panic("unable to find compatible ram conroller node in dtb\n"); + +	at91_ramc_base[0] = of_iomap(np, 0); +	if (!at91_ramc_base[0]) +		panic("unable to map ramc[0] cpu registers\n"); +	/* the controller may have 2 banks */ +	at91_ramc_base[1] = of_iomap(np, 1); + +	of_node_put(np); +} + +static struct of_device_id shdwc_ids[] = { +	{ .compatible = "atmel,at91sam9260-shdwc", }, +	{ .compatible = "atmel,at91sam9rl-shdwc", }, +	{ .compatible = "atmel,at91sam9x5-shdwc", }, +	{ /*sentinel*/ } +}; + +static const char *shdwc_wakeup_modes[] = { +	[AT91_SHDW_WKMODE0_NONE]	= "none", +	[AT91_SHDW_WKMODE0_HIGH]	= "high", +	[AT91_SHDW_WKMODE0_LOW]		= "low", +	[AT91_SHDW_WKMODE0_ANYLEVEL]	= "any", +}; + +const int at91_dtget_shdwc_wakeup_mode(struct device_node *np) +{ +	const char *pm; +	int err, i; + +	err = of_property_read_string(np, "atmel,wakeup-mode", &pm); +	if (err < 0) +		return AT91_SHDW_WKMODE0_ANYLEVEL; + +	for (i = 0; i < ARRAY_SIZE(shdwc_wakeup_modes); i++) +		if (!strcasecmp(pm, shdwc_wakeup_modes[i])) +			return i; + +	return -ENODEV; +} + +static void at91_dt_shdwc(void) +{ +	struct device_node *np; +	int wakeup_mode; +	u32 reg; +	u32 mode = 0; + +	np = of_find_matching_node(NULL, shdwc_ids); +	if (!np) { +		pr_debug("AT91: unable to find compatible shutdown (shdwc) conroller node in dtb\n"); +		return; +	} + +	at91_shdwc_base = of_iomap(np, 0); +	if (!at91_shdwc_base) +		panic("AT91: unable to map shdwc cpu registers\n"); + +	wakeup_mode = at91_dtget_shdwc_wakeup_mode(np); +	if (wakeup_mode < 0) { +		pr_warn("AT91: shdwc unknown wakeup mode\n"); +		goto end; +	} + +	if (!of_property_read_u32(np, "atmel,wakeup-counter", ®)) { +		if (reg > AT91_SHDW_CPTWK0_MAX) { +			pr_warn("AT91: shdwc wakeup conter 0x%x > 0x%x reduce it to 0x%x\n", +				reg, AT91_SHDW_CPTWK0_MAX, AT91_SHDW_CPTWK0_MAX); +			reg = AT91_SHDW_CPTWK0_MAX; +		} +		mode |= AT91_SHDW_CPTWK0_(reg); +	} + +	if (of_property_read_bool(np, "atmel,wakeup-rtc-timer")) +			mode |= AT91_SHDW_RTCWKEN; + +	if (of_property_read_bool(np, "atmel,wakeup-rtt-timer")) +			mode |= AT91_SHDW_RTTWKEN; + +	at91_shdwc_write(AT91_SHDW_MR, wakeup_mode | mode); + +end: +	pm_power_off = at91sam9_poweroff; + +	of_node_put(np); +} + +void __init at91_dt_initialize(void) +{ +	at91_dt_rstc(); +	at91_dt_ramc(); +	at91_dt_shdwc(); + +	/* Init clock subsystem */ +	at91_dt_clock_init(); + +	/* Register the processor-specific clocks */ +	at91_boot_soc.register_clocks(); + +	at91_boot_soc.init(); +} +#endif +  void __init at91_initialize(unsigned long main_clock)  {  	at91_boot_soc.ioremap_registers();  |