Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit cf0ce095 authored by Linus Walleij's avatar Linus Walleij
Browse files

ARM: u300: add syscon node



This adds a device tree node for the U300 system controller
and remaps this dynamically instead of using hard-coded
virtual addresses. The board power set-up code is altered
to fetch a reference to the syscon using ampersand <&syscon>
notation. This way of passing a pointer to the syscon will
also be used by the clocks.

Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent be2885a5
Loading
Loading
Loading
Loading
+27 −3
Original line number Original line Diff line number Diff line
@@ -8,15 +8,39 @@ Required root node property:


compatible="stericsson,u300";
compatible="stericsson,u300";


Required node: syscon
This contains the system controller.
- compatible: must be "stericsson,u300-syscon".
- reg: the base address and size of the system controller.

Boards with the U300 SoC include:
Boards with the U300 SoC include:


S365 "Small Board U365":
S365 "Small Board U365":


Required node: s365
Required node: s365
This contains the board-specific information.
- compatible: must be "stericsson,s365".
- vana15-supply: the regulator supplying the 1.5V to drive the
  board.
- syscon: a pointer to the syscon node so we can acccess the
  syscon registers to set the board as self-powered.


Example:
Example:


/ {
	model = "ST-Ericsson U300";
	compatible = "stericsson,u300";
	#address-cells = <1>;
	#size-cells = <1>;

	s365 {
	s365 {
		compatible = "stericsson,s365";
		compatible = "stericsson,s365";
		vana15-supply = <&ab3100_ldo_d_reg>;
		vana15-supply = <&ab3100_ldo_d_reg>;
		syscon = <&syscon>;
	};

	syscon: syscon@c0011000 {
		compatible = "stericsson,u300-syscon";
		reg = <0xc0011000 0x1000>;
	};
};
};
+6 −0
Original line number Original line Diff line number Diff line
@@ -27,6 +27,12 @@
	s365 {
	s365 {
		compatible = "stericsson,s365";
		compatible = "stericsson,s365";
		vana15-supply = <&ab3100_ldo_d_reg>;
		vana15-supply = <&ab3100_ldo_d_reg>;
		syscon = <&syscon>;
	};

	syscon: syscon@c0011000 {
		compatible = "stericsson,u300-syscon";
		reg = <0xc0011000 0x1000>;
	};
	};


	timer: timer@c0014000 {
	timer: timer@c0014000 {
+18 −4
Original line number Original line Diff line number Diff line
@@ -14,6 +14,7 @@
#include <linux/platform_data/clk-u300.h>
#include <linux/platform_data/clk-u300.h>
#include <linux/platform_data/pinctrl-coh901.h>
#include <linux/platform_data/pinctrl-coh901.h>
#include <linux/irqchip.h>
#include <linux/irqchip.h>
#include <linux/of_address.h>
#include <linux/of_platform.h>
#include <linux/of_platform.h>
#include <linux/clocksource.h>
#include <linux/clocksource.h>
#include <linux/clk.h>
#include <linux/clk.h>
@@ -48,6 +49,8 @@
#define U300_SYSCON_BCR_EXTRA_BOOT_OPTION_MASK			(0x01FC)
#define U300_SYSCON_BCR_EXTRA_BOOT_OPTION_MASK			(0x01FC)
#define U300_SYSCON_BCR_APP_BOOT_SERV_MASK			(0x0003)
#define U300_SYSCON_BCR_APP_BOOT_SERV_MASK			(0x0003)


static void __iomem *syscon_base;

/*
/*
 * Static I/O mappings that are needed for booting the U300 platforms. The
 * Static I/O mappings that are needed for booting the U300 platforms. The
 * only things we need are the areas where we find the timer, syscon and
 * only things we need are the areas where we find the timer, syscon and
@@ -171,7 +174,7 @@ static void __init u300_init_check_chip(void)
	const char unknown[] = "UNKNOWN";
	const char unknown[] = "UNKNOWN";


	/* Read out and print chip ID */
	/* Read out and print chip ID */
	val = readw(U300_SYSCON_VBASE + U300_SYSCON_CIDR);
	val = readw(syscon_base + U300_SYSCON_CIDR);
	/* This is in funky bigendian order... */
	/* This is in funky bigendian order... */
	val = (val & 0xFFU) << 8 | (val >> 8);
	val = (val & 0xFFU) << 8 | (val >> 8);
	chip = db_chips;
	chip = db_chips;
@@ -244,10 +247,21 @@ static struct of_dev_auxdata u300_auxdata_lookup[] __initdata = {


static void __init u300_init_irq_dt(void)
static void __init u300_init_irq_dt(void)
{
{
	struct device_node *syscon;
	struct clk *clk;
	struct clk *clk;


	syscon = of_find_node_by_path("/syscon@c0011000");
	if (!syscon) {
		pr_crit("could not find syscon node\n");
		return;
	}
	syscon_base = of_iomap(syscon, 0);
	if (!syscon_base) {
		pr_crit("could not remap syscon\n");
		return;
	}
	/* initialize clocking early, we want to clock the INTCON */
	/* initialize clocking early, we want to clock the INTCON */
	u300_clk_init(U300_SYSCON_VBASE);
	u300_clk_init(syscon_base);


	/* Bootstrap EMIF and SEMI clocks */
	/* Bootstrap EMIF and SEMI clocks */
	clk = clk_get_sys("pl172", NULL);
	clk = clk_get_sys("pl172", NULL);
@@ -280,9 +294,9 @@ static void __init u300_init_machine_dt(void)
			u300_auxdata_lookup, NULL);
			u300_auxdata_lookup, NULL);


	/* Enable SEMI self refresh */
	/* Enable SEMI self refresh */
	val = readw(U300_SYSCON_VBASE + U300_SYSCON_SMCR) |
	val = readw(syscon_base + U300_SYSCON_SMCR) |
		U300_SYSCON_SMCR_SEMI_SREFREQ_ENABLE;
		U300_SYSCON_SMCR_SEMI_SREFREQ_ENABLE;
	writew(val, U300_SYSCON_VBASE + U300_SYSCON_SMCR);
	writew(val, syscon_base + U300_SYSCON_SMCR);
}
}


static const char * u300_board_compat[] = {
static const char * u300_board_compat[] = {
+18 −3
Original line number Original line Diff line number Diff line
@@ -16,8 +16,8 @@
#include <linux/regulator/machine.h>
#include <linux/regulator/machine.h>
#include <linux/regulator/consumer.h>
#include <linux/regulator/consumer.h>
/* Those are just for writing in syscon */
/* Those are just for writing in syscon */
#include <linux/of_address.h>
#include <linux/io.h>
#include <linux/io.h>
#include "u300-regs.h"


/* Power Management Control 16bit (R/W) */
/* Power Management Control 16bit (R/W) */
#define U300_SYSCON_PMCR					(0x50)
#define U300_SYSCON_PMCR					(0x50)
@@ -57,10 +57,25 @@ void u300_pm_poweroff(void)
 */
 */
static int __init __u300_init_boardpower(struct platform_device *pdev)
static int __init __u300_init_boardpower(struct platform_device *pdev)
{
{
	struct device_node *np = pdev->dev.of_node;
	struct device_node *syscon_np;
	static void __iomem *syscon_base;
	int err;
	int err;
	u32 val;
	u32 val;


	pr_info("U300: setting up board power\n");
	pr_info("U300: setting up board power\n");

	syscon_np = of_parse_phandle(np, "syscon", 0);
	if (!syscon_np) {
		pr_crit("U300: no syscon node\n");
		return -ENODEV;
	}
	syscon_base = of_iomap(syscon_np, 0);
	if (!syscon_base) {
		pr_crit("U300: could not remap syscon\n");
		return -ENODEV;
	}

	main_power_15 = regulator_get(&pdev->dev, "vana15");
	main_power_15 = regulator_get(&pdev->dev, "vana15");


	if (IS_ERR(main_power_15)) {
	if (IS_ERR(main_power_15)) {
@@ -81,9 +96,9 @@ static int __init __u300_init_boardpower(struct platform_device *pdev)
	 * the rest of the U300 power management is implemented.
	 * the rest of the U300 power management is implemented.
	 */
	 */
	pr_info("U300: disable system controller pull-up\n");
	pr_info("U300: disable system controller pull-up\n");
	val = readw(U300_SYSCON_VBASE + U300_SYSCON_PMCR);
	val = readw(syscon_base + U300_SYSCON_PMCR);
	val &= ~U300_SYSCON_PMCR_DCON_ENABLE;
	val &= ~U300_SYSCON_PMCR_DCON_ENABLE;
	writew(val, U300_SYSCON_VBASE + U300_SYSCON_PMCR);
	writew(val, syscon_base + U300_SYSCON_PMCR);


	/* Register globally exported PM poweroff hook */
	/* Register globally exported PM poweroff hook */
	pm_power_off = u300_pm_poweroff;
	pm_power_off = u300_pm_poweroff;