Message ID | 1406842092-25207-2-git-send-email-emilio@elopez.com.ar (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Hi Emilio, On Thu, Jul 31, 2014 at 06:28:04PM -0300, Emilio López wrote: > This commit adds SoC bus support on the sunxi platform, and exposes > information such as the hardware revision to userspace and other kernel > clients during init. A message with this information is also printed to > the kernel log to ease future bug triaging. > > Signed-off-by: Emilio López <emilio@elopez.com.ar> > --- > arch/arm/mach-sunxi/Kconfig | 1 + > arch/arm/mach-sunxi/Makefile | 2 +- > arch/arm/mach-sunxi/sunxi-soc-id.c | 226 +++++++++++++++++++++++++++++++++++++ > arch/arm/mach-sunxi/sunxi-soc-id.h | 6 + > 4 files changed, 234 insertions(+), 1 deletion(-) > create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.c > create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.h > > diff --git a/arch/arm/mach-sunxi/Kconfig b/arch/arm/mach-sunxi/Kconfig > index 6434e3b..4a199df 100644 > --- a/arch/arm/mach-sunxi/Kconfig > +++ b/arch/arm/mach-sunxi/Kconfig > @@ -5,6 +5,7 @@ menuconfig ARCH_SUNXI > select GENERIC_IRQ_CHIP > select PINCTRL > select PINCTRL_SUNXI > + select SOC_BUS > select SUN4I_TIMER > > if ARCH_SUNXI > diff --git a/arch/arm/mach-sunxi/Makefile b/arch/arm/mach-sunxi/Makefile > index 27b168f..589239b 100644 > --- a/arch/arm/mach-sunxi/Makefile > +++ b/arch/arm/mach-sunxi/Makefile > @@ -1,2 +1,2 @@ > -obj-$(CONFIG_ARCH_SUNXI) += sunxi.o > +obj-$(CONFIG_ARCH_SUNXI) += sunxi.o sunxi-soc-id.o > obj-$(CONFIG_SMP) += platsmp.o > diff --git a/arch/arm/mach-sunxi/sunxi-soc-id.c b/arch/arm/mach-sunxi/sunxi-soc-id.c > new file mode 100644 > index 0000000..c7eff1c > --- /dev/null > +++ b/arch/arm/mach-sunxi/sunxi-soc-id.c > @@ -0,0 +1,226 @@ > +/* > + * SoC revision detection for sunxi SoCs > + * > + * Copyright 2014 Emilio López > + * > + * Emilio López <emilio@elopez.com.ar> > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License as published by > + * the Free Software Foundation; either version 2 of the License, or > + * (at your option) any later version. > + * > + * This program is distributed in the hope that it will be useful, > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > + * GNU General Public License for more details. > + */ > + > +#include <linux/err.h> > +#include <linux/io.h> > +#include <linux/of.h> > +#include <linux/of_address.h> > +#include <linux/slab.h> > +#include <linux/string.h> > +#include <linux/sys_soc.h> > + > +#include "sunxi-soc-id.h" > + > +/* > + * On the A10 SoC, we can read the revision information from the timer > + * block. The detection logic is extracted from similar code on the > + * Allwinner vendor tree, as this is undocumented on the user manual > + */ > + > +#define TIMER_SOC_REV_REG 0x13c > +#define TIMER_SOC_REV_CLEAR(val) ((val) & ~(0x3 << 6)) > +#define TIMER_SOC_REV_GET(val) (((val) >> 6) & 0x3) > + > +static const struct of_device_id sun4i_timer_compatible[] __initconst = { > + { .compatible = "allwinner,sun4i-a10-timer", }, > + {}, > +}; > + > +static int __init sun4i_get_soc_revision(void) > +{ > + struct device_node *np; > + void __iomem *base; > + u32 val; > + int ret; > + > + /* Find the timer node */ > + np = of_find_matching_node(NULL, sun4i_timer_compatible); > + if (!np) > + return -ENODEV; > + > + /* Temporarily map it for reading */ > + base = of_iomap(np, 0); > + if (!base) { > + of_node_put(np); > + return -ENOMEM; > + } > + > + /* Clear the SoC revision bits and rewrite the register */ > + val = readl(base + TIMER_SOC_REV_REG); > + val = TIMER_SOC_REV_CLEAR(val); > + writel(val, base + TIMER_SOC_REV_REG); > + > + /* Now read it again and see what shows up */ > + val = readl(base + TIMER_SOC_REV_REG); > + val = TIMER_SOC_REV_GET(val); > + > + switch (val) { > + case 0: /* revision A */ > + ret = 'A'; > + case 3: /* revision B */ > + ret = 'B'; > + default: /* revision C */ > + ret = 'C'; What's programmed in there in the case of the rev C? > + } > + > + iounmap(base); > + of_node_put(np); > + > + return ret; > +} > + > +/* > + * On the sun5i SoCs (A10S, A13), we can read the revision information > + * from the first bits in the Security ID. The detection logic is > + * extracted from similar code on the Allwinner vendor tree, as this > + * is undocumented on the user manual. > + */ > + > +static const struct of_device_id sun5i_sid_compatible[] __initconst = { > + { .compatible = "allwinner,sun4i-a10-sid", }, > + {}, > +}; > + > +static int __init sun5i_get_soc_revision(void) > +{ > + struct device_node *np; > + void __iomem *sid; > + u32 val; > + int ret; > + > + /* Find the SID node */ > + np = of_find_matching_node(NULL, sun5i_sid_compatible); > + if (!np) > + return -ENODEV; > + > + /* Temporarily map it for reading */ > + sid = of_iomap(np, 0); > + if (!sid) { > + of_node_put(np); > + return -ENOMEM; > + } > + > + /* Read and extract the chip revision from the SID */ > + val = readl(sid); > + val = (val >> 8) & 0xffffff; > + > + switch (val) { > + case 0: /* A10S/A13 rev A */ > + case 0x162541: /* A10S/A13 rev A */ > + case 0x162565: /* A13 rev A */ Some defines would be nice. Also, isn't the SID supposed to identify any SoC, not just the sun5i? > + ret = 'A'; > + break; > + case 0x162542: /* A10S/A13 rev B */ > + ret = 'B'; > + break; > + default: /* Unknown chip revision */ > + ret = -ENODATA; > + } > + > + iounmap(sid); > + of_node_put(np); > + > + return ret; > +} > + > +int __init sunxi_soc_revision(void) > +{ > + static int revision = -ENODEV; > + > + /* Try to query the hardware just once */ > + if (!IS_ERR_VALUE(revision)) > + return revision; > + > + if (of_machine_is_compatible("allwinner,sun4i-a10")) { > + revision = sun4i_get_soc_revision(); > + } else if (of_machine_is_compatible("allwinner,sun5i-a10s") || > + of_machine_is_compatible("allwinner,sun5i-a13")) { > + revision = sun5i_get_soc_revision(); > + } > + > + return revision; > +} > + > +/* Matches for the sunxi SoCs we know of */ > +static const struct of_device_id soc_matches[] __initconst = { > + { .compatible = "allwinner,sun4i-a10", .data = "A10 (sun4i)" }, > + { .compatible = "allwinner,sun5i-a13", .data = "A13 (sun5i)" }, > + { .compatible = "allwinner,sun5i-a10s", .data = "A10S (sun5i)" }, > + { .compatible = "allwinner,sun6i-a31", .data = "A31 (sun6i)" }, > + { .compatible = "allwinner,sun7i-a20", .data = "A20 (sun7i)" }, > + { .compatible = "allwinner,sun8i-a23", .data = "A23 (sun8i)" }, > + { /* sentinel */ }, > +}; Hmmm, no, either auto-detect it, or don't, but that's useless. We can already have the same information from the DT directly. > +static int __init sunxi_register_soc_device(void) > +{ > + struct soc_device_attribute *soc_dev_attr; > + struct soc_device *soc_dev; > + const struct of_device_id *match; > + struct device_node *root; > + int revision; > + > + /* Only run on sunxi SoCs that we know of */ > + root = of_find_node_by_path("/"); > + match = of_match_node(soc_matches, root); > + if (!match) > + goto exit; > + > + soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL); > + if (!soc_dev_attr) > + goto exit; > + > + /* Read the machine name if available */ > + of_property_read_string(root, "model", &soc_dev_attr->machine); > + > + soc_dev_attr->family = kstrdup("Allwinner A Series", GFP_KERNEL); I think the family should be sun*i > + soc_dev_attr->soc_id = kstrdup(match->data, GFP_KERNEL); And soc_id would be just the name of the SoC. > + > + /* Revision may not always be available */ > + revision = sunxi_soc_revision(); > + if (IS_ERR_VALUE(revision)) > + soc_dev_attr->revision = kstrdup("Unknown", GFP_KERNEL); > + else > + soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%c", revision); > + > + soc_dev = soc_device_register(soc_dev_attr); > + if (IS_ERR(soc_dev)) > + goto free_struct; > + > + /* > + * Print an informational line mentioning the hardware details. > + * It may come in handy during bug reports, as some early SoC > + * revisions have hardware quirks and do not get much testing. > + */ > + pr_info("SoC bus registered, running %s %s, revision %s\n", > + soc_dev_attr->family, soc_dev_attr->soc_id, > + soc_dev_attr->revision); Maybe a pr_fmt would be nice here. > + > + return 0; > + > +free_struct: > + kfree(soc_dev_attr->family); > + kfree(soc_dev_attr->soc_id); > + kfree(soc_dev_attr->revision); > + kfree(soc_dev_attr); > +exit: > + of_node_put(root); > + > + return 0; > +} > +postcore_initcall(sunxi_register_soc_device) I'm kind of reluctant to this. That would mean that it will run for every SoC. Can't you tie it to the system controller, and have it probed as usual?
Hi Maxime, Thanks for taking the time to review this :) El 03/08/14 a las 09:40, Maxime Ripard escibió: > Hi Emilio, > > On Thu, Jul 31, 2014 at 06:28:04PM -0300, Emilio López wrote: >> This commit adds SoC bus support on the sunxi platform, and exposes >> information such as the hardware revision to userspace and other kernel >> clients during init. A message with this information is also printed to >> the kernel log to ease future bug triaging. >> >> Signed-off-by: Emilio López <emilio@elopez.com.ar> >> --- >> arch/arm/mach-sunxi/Kconfig | 1 + >> arch/arm/mach-sunxi/Makefile | 2 +- >> arch/arm/mach-sunxi/sunxi-soc-id.c | 226 +++++++++++++++++++++++++++++++++++++ >> arch/arm/mach-sunxi/sunxi-soc-id.h | 6 + >> 4 files changed, 234 insertions(+), 1 deletion(-) >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.c >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.h >> >> diff --git a/arch/arm/mach-sunxi/Kconfig b/arch/arm/mach-sunxi/Kconfig >> index 6434e3b..4a199df 100644 >> --- a/arch/arm/mach-sunxi/Kconfig >> +++ b/arch/arm/mach-sunxi/Kconfig >> @@ -5,6 +5,7 @@ menuconfig ARCH_SUNXI >> select GENERIC_IRQ_CHIP >> select PINCTRL >> select PINCTRL_SUNXI >> + select SOC_BUS >> select SUN4I_TIMER >> >> if ARCH_SUNXI >> diff --git a/arch/arm/mach-sunxi/Makefile b/arch/arm/mach-sunxi/Makefile >> index 27b168f..589239b 100644 >> --- a/arch/arm/mach-sunxi/Makefile >> +++ b/arch/arm/mach-sunxi/Makefile >> @@ -1,2 +1,2 @@ >> -obj-$(CONFIG_ARCH_SUNXI) += sunxi.o >> +obj-$(CONFIG_ARCH_SUNXI) += sunxi.o sunxi-soc-id.o >> obj-$(CONFIG_SMP) += platsmp.o >> diff --git a/arch/arm/mach-sunxi/sunxi-soc-id.c b/arch/arm/mach-sunxi/sunxi-soc-id.c >> new file mode 100644 >> index 0000000..c7eff1c >> --- /dev/null >> +++ b/arch/arm/mach-sunxi/sunxi-soc-id.c >> @@ -0,0 +1,226 @@ >> +/* >> + * SoC revision detection for sunxi SoCs >> + * >> + * Copyright 2014 Emilio López >> + * >> + * Emilio López <emilio@elopez.com.ar> >> + * >> + * This program is free software; you can redistribute it and/or modify >> + * it under the terms of the GNU General Public License as published by >> + * the Free Software Foundation; either version 2 of the License, or >> + * (at your option) any later version. >> + * >> + * This program is distributed in the hope that it will be useful, >> + * but WITHOUT ANY WARRANTY; without even the implied warranty of >> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + * GNU General Public License for more details. >> + */ >> + >> +#include <linux/err.h> >> +#include <linux/io.h> >> +#include <linux/of.h> >> +#include <linux/of_address.h> >> +#include <linux/slab.h> >> +#include <linux/string.h> >> +#include <linux/sys_soc.h> >> + >> +#include "sunxi-soc-id.h" >> + >> +/* >> + * On the A10 SoC, we can read the revision information from the timer >> + * block. The detection logic is extracted from similar code on the >> + * Allwinner vendor tree, as this is undocumented on the user manual >> + */ >> + >> +#define TIMER_SOC_REV_REG 0x13c >> +#define TIMER_SOC_REV_CLEAR(val) ((val) & ~(0x3 << 6)) >> +#define TIMER_SOC_REV_GET(val) (((val) >> 6) & 0x3) >> + >> +static const struct of_device_id sun4i_timer_compatible[] __initconst = { >> + { .compatible = "allwinner,sun4i-a10-timer", }, >> + {}, >> +}; >> + >> +static int __init sun4i_get_soc_revision(void) >> +{ >> + struct device_node *np; >> + void __iomem *base; >> + u32 val; >> + int ret; >> + >> + /* Find the timer node */ >> + np = of_find_matching_node(NULL, sun4i_timer_compatible); >> + if (!np) >> + return -ENODEV; >> + >> + /* Temporarily map it for reading */ >> + base = of_iomap(np, 0); >> + if (!base) { >> + of_node_put(np); >> + return -ENOMEM; >> + } >> + >> + /* Clear the SoC revision bits and rewrite the register */ >> + val = readl(base + TIMER_SOC_REV_REG); >> + val = TIMER_SOC_REV_CLEAR(val); >> + writel(val, base + TIMER_SOC_REV_REG); >> + >> + /* Now read it again and see what shows up */ >> + val = readl(base + TIMER_SOC_REV_REG); >> + val = TIMER_SOC_REV_GET(val); >> + >> + switch (val) { >> + case 0: /* revision A */ >> + ret = 'A'; >> + case 3: /* revision B */ >> + ret = 'B'; >> + default: /* revision C */ >> + ret = 'C'; > > What's programmed in there in the case of the rev C? Something that's not a 3 or a 0 - I don't have any more specifics than what the Allwinner code does :( https://github.com/linux-sunxi/linux-sunxi/blob/lichee-dev/arch/arm/mach-sun4i/core.c#L418 I just noticed that I forgot the breaks while reworking this from return to a variable, so everything is rev C right now - I'll fix that. >> + } >> + >> + iounmap(base); >> + of_node_put(np); >> + >> + return ret; >> +} >> + >> +/* >> + * On the sun5i SoCs (A10S, A13), we can read the revision information >> + * from the first bits in the Security ID. The detection logic is >> + * extracted from similar code on the Allwinner vendor tree, as this >> + * is undocumented on the user manual. >> + */ >> + >> +static const struct of_device_id sun5i_sid_compatible[] __initconst = { >> + { .compatible = "allwinner,sun4i-a10-sid", }, >> + {}, >> +}; >> + >> +static int __init sun5i_get_soc_revision(void) >> +{ >> + struct device_node *np; >> + void __iomem *sid; >> + u32 val; >> + int ret; >> + >> + /* Find the SID node */ >> + np = of_find_matching_node(NULL, sun5i_sid_compatible); >> + if (!np) >> + return -ENODEV; >> + >> + /* Temporarily map it for reading */ >> + sid = of_iomap(np, 0); >> + if (!sid) { >> + of_node_put(np); >> + return -ENOMEM; >> + } >> + >> + /* Read and extract the chip revision from the SID */ >> + val = readl(sid); >> + val = (val >> 8) & 0xffffff; >> + >> + switch (val) { >> + case 0: /* A10S/A13 rev A */ >> + case 0x162541: /* A10S/A13 rev A */ >> + case 0x162565: /* A13 rev A */ > > Some defines would be nice. Ok > Also, isn't the SID supposed to identify any SoC, not just the sun5i? The Allwinner code I've seen uses the SS to detect the SoC (which luckily we don't need, as we have DT) and then reads the timer for sun4i or the SID for sun5i. >> + ret = 'A'; >> + break; >> + case 0x162542: /* A10S/A13 rev B */ >> + ret = 'B'; >> + break; >> + default: /* Unknown chip revision */ >> + ret = -ENODATA; >> + } >> + >> + iounmap(sid); >> + of_node_put(np); >> + >> + return ret; >> +} >> + >> +int __init sunxi_soc_revision(void) >> +{ >> + static int revision = -ENODEV; >> + >> + /* Try to query the hardware just once */ >> + if (!IS_ERR_VALUE(revision)) >> + return revision; >> + >> + if (of_machine_is_compatible("allwinner,sun4i-a10")) { >> + revision = sun4i_get_soc_revision(); >> + } else if (of_machine_is_compatible("allwinner,sun5i-a10s") || >> + of_machine_is_compatible("allwinner,sun5i-a13")) { >> + revision = sun5i_get_soc_revision(); >> + } >> + >> + return revision; >> +} >> + >> +/* Matches for the sunxi SoCs we know of */ >> +static const struct of_device_id soc_matches[] __initconst = { >> + { .compatible = "allwinner,sun4i-a10", .data = "A10 (sun4i)" }, >> + { .compatible = "allwinner,sun5i-a13", .data = "A13 (sun5i)" }, >> + { .compatible = "allwinner,sun5i-a10s", .data = "A10S (sun5i)" }, >> + { .compatible = "allwinner,sun6i-a31", .data = "A31 (sun6i)" }, >> + { .compatible = "allwinner,sun7i-a20", .data = "A20 (sun7i)" }, >> + { .compatible = "allwinner,sun8i-a23", .data = "A23 (sun8i)" }, >> + { /* sentinel */ }, >> +}; > > Hmmm, no, either auto-detect it, or don't, but that's useless. We can > already have the same information from the DT directly. I don't quite get what you're trying to say here. This is here to a) get a fancy string saying what the SoC is (AFAIK that's not directly available on the DT) b) only run in the listed SoCs >> +static int __init sunxi_register_soc_device(void) >> +{ >> + struct soc_device_attribute *soc_dev_attr; >> + struct soc_device *soc_dev; >> + const struct of_device_id *match; >> + struct device_node *root; >> + int revision; >> + >> + /* Only run on sunxi SoCs that we know of */ >> + root = of_find_node_by_path("/"); >> + match = of_match_node(soc_matches, root); >> + if (!match) >> + goto exit; See here >> + >> + soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL); >> + if (!soc_dev_attr) >> + goto exit; >> + >> + /* Read the machine name if available */ >> + of_property_read_string(root, "model", &soc_dev_attr->machine); >> + >> + soc_dev_attr->family = kstrdup("Allwinner A Series", GFP_KERNEL); > > I think the family should be sun*i Here is a list of the ones currently in use for family "Cirrus Logic EP93xx" "Freescale i.MX" "Integrator" "Marvell" "Freescale MXS Family" "Tegra" "Xilinx Zynq" There does not seem to be a real consensus on what these mean. There's a binding document on Documentation/ABI/testing/sysfs-devices-soc but it's not really detailed. > >> + soc_dev_attr->soc_id = kstrdup(match->data, GFP_KERNEL); > > And soc_id would be just the name of the SoC. According to the binding this is a serial number. Lee, could you help us decide what these fields should look like? I'm now inclining to have eg. family=sun5i, machine=A13, keep revision the same and drop soc_id. >> + >> + /* Revision may not always be available */ >> + revision = sunxi_soc_revision(); >> + if (IS_ERR_VALUE(revision)) >> + soc_dev_attr->revision = kstrdup("Unknown", GFP_KERNEL); >> + else >> + soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%c", revision); >> + >> + soc_dev = soc_device_register(soc_dev_attr); >> + if (IS_ERR(soc_dev)) >> + goto free_struct; >> + >> + /* >> + * Print an informational line mentioning the hardware details. >> + * It may come in handy during bug reports, as some early SoC >> + * revisions have hardware quirks and do not get much testing. >> + */ >> + pr_info("SoC bus registered, running %s %s, revision %s\n", >> + soc_dev_attr->family, soc_dev_attr->soc_id, >> + soc_dev_attr->revision); > > Maybe a pr_fmt would be nice here. With something like "Allwinner SoC bus"? >> + >> + return 0; >> + >> +free_struct: >> + kfree(soc_dev_attr->family); >> + kfree(soc_dev_attr->soc_id); >> + kfree(soc_dev_attr->revision); >> + kfree(soc_dev_attr); >> +exit: >> + of_node_put(root); >> + >> + return 0; >> +} >> +postcore_initcall(sunxi_register_soc_device) > > I'm kind of reluctant to this. That would mean that it will run for > every SoC. The overhead should be negligible - on non-sunxi, it will only check if the machine is compatible by looking at the root node and bail out as it's not the case. > Can't you tie it to the system controller, and have it probed as > usual? I considered tying this to init_machine, right before the DT starts to get probed. That would mean actually writing init_machine hooks calling into this for some of our machines, and getting us further from the generic DT machine nirvana though. As the quirk bits do not directly depend on this, we can probably get away with making this a normal driver compatible with all our SoCs. Let me know which way do you prefer, and I'll be happy to oblige. Cheers! Emilio
On Sun, 03 Aug 2014, Emilio López wrote: > El 03/08/14 a las 09:40, Maxime Ripard escibió: > >On Thu, Jul 31, 2014 at 06:28:04PM -0300, Emilio López wrote: > >>This commit adds SoC bus support on the sunxi platform, and exposes > >>information such as the hardware revision to userspace and other kernel > >>clients during init. A message with this information is also printed to > >>the kernel log to ease future bug triaging. > >> > >>Signed-off-by: Emilio López <emilio@elopez.com.ar> > >>--- > >> arch/arm/mach-sunxi/Kconfig | 1 + > >> arch/arm/mach-sunxi/Makefile | 2 +- > >> arch/arm/mach-sunxi/sunxi-soc-id.c | 226 +++++++++++++++++++++++++++++++++++++ > >> arch/arm/mach-sunxi/sunxi-soc-id.h | 6 + > >> 4 files changed, 234 insertions(+), 1 deletion(-) > >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.c > >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.h [...] > >>+ soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL); > >>+ if (!soc_dev_attr) > >>+ goto exit; > >>+ > >>+ /* Read the machine name if available */ > >>+ of_property_read_string(root, "model", &soc_dev_attr->machine); > >>+ > >>+ soc_dev_attr->family = kstrdup("Allwinner A Series", GFP_KERNEL); > > > >I think the family should be sun*i > > Here is a list of the ones currently in use for family > > "Cirrus Logic EP93xx" > "Freescale i.MX" > "Integrator" > "Marvell" > "Freescale MXS Family" > "Tegra" > "Xilinx Zynq" > > There does not seem to be a real consensus on what these mean. > There's a binding document on > Documentation/ABI/testing/sysfs-devices-soc but it's not really > detailed. > > > > >>+ soc_dev_attr->soc_id = kstrdup(match->data, GFP_KERNEL); > > > >And soc_id would be just the name of the SoC. > > According to the binding this is a serial number. Lee, could you > help us decide what these fields should look like? > > I'm now inclining to have eg. family=sun5i, machine=A13, keep > revision the same and drop soc_id. Voila: Documentation/ABI/testing/sysfs-devices-soc
Hi, On Sun, Aug 03, 2014 at 06:45:37PM -0300, Emilio López wrote: > Hi Maxime, > > Thanks for taking the time to review this :) You're welcome > > El 03/08/14 a las 09:40, Maxime Ripard escibió: > >Hi Emilio, > > > >On Thu, Jul 31, 2014 at 06:28:04PM -0300, Emilio López wrote: > >>This commit adds SoC bus support on the sunxi platform, and exposes > >>information such as the hardware revision to userspace and other kernel > >>clients during init. A message with this information is also printed to > >>the kernel log to ease future bug triaging. > >> > >>Signed-off-by: Emilio López <emilio@elopez.com.ar> > >>--- > >> arch/arm/mach-sunxi/Kconfig | 1 + > >> arch/arm/mach-sunxi/Makefile | 2 +- > >> arch/arm/mach-sunxi/sunxi-soc-id.c | 226 +++++++++++++++++++++++++++++++++++++ > >> arch/arm/mach-sunxi/sunxi-soc-id.h | 6 + > >> 4 files changed, 234 insertions(+), 1 deletion(-) > >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.c > >> create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.h > >> > >>diff --git a/arch/arm/mach-sunxi/Kconfig b/arch/arm/mach-sunxi/Kconfig > >>index 6434e3b..4a199df 100644 > >>--- a/arch/arm/mach-sunxi/Kconfig > >>+++ b/arch/arm/mach-sunxi/Kconfig > >>@@ -5,6 +5,7 @@ menuconfig ARCH_SUNXI > >> select GENERIC_IRQ_CHIP > >> select PINCTRL > >> select PINCTRL_SUNXI > >>+ select SOC_BUS > >> select SUN4I_TIMER > >> > >> if ARCH_SUNXI > >>diff --git a/arch/arm/mach-sunxi/Makefile b/arch/arm/mach-sunxi/Makefile > >>index 27b168f..589239b 100644 > >>--- a/arch/arm/mach-sunxi/Makefile > >>+++ b/arch/arm/mach-sunxi/Makefile > >>@@ -1,2 +1,2 @@ > >>-obj-$(CONFIG_ARCH_SUNXI) += sunxi.o > >>+obj-$(CONFIG_ARCH_SUNXI) += sunxi.o sunxi-soc-id.o > >> obj-$(CONFIG_SMP) += platsmp.o > >>diff --git a/arch/arm/mach-sunxi/sunxi-soc-id.c b/arch/arm/mach-sunxi/sunxi-soc-id.c > >>new file mode 100644 > >>index 0000000..c7eff1c > >>--- /dev/null > >>+++ b/arch/arm/mach-sunxi/sunxi-soc-id.c > >>@@ -0,0 +1,226 @@ > >>+/* > >>+ * SoC revision detection for sunxi SoCs > >>+ * > >>+ * Copyright 2014 Emilio López > >>+ * > >>+ * Emilio López <emilio@elopez.com.ar> > >>+ * > >>+ * This program is free software; you can redistribute it and/or modify > >>+ * it under the terms of the GNU General Public License as published by > >>+ * the Free Software Foundation; either version 2 of the License, or > >>+ * (at your option) any later version. > >>+ * > >>+ * This program is distributed in the hope that it will be useful, > >>+ * but WITHOUT ANY WARRANTY; without even the implied warranty of > >>+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > >>+ * GNU General Public License for more details. > >>+ */ > >>+ > >>+#include <linux/err.h> > >>+#include <linux/io.h> > >>+#include <linux/of.h> > >>+#include <linux/of_address.h> > >>+#include <linux/slab.h> > >>+#include <linux/string.h> > >>+#include <linux/sys_soc.h> > >>+ > >>+#include "sunxi-soc-id.h" > >>+ > >>+/* > >>+ * On the A10 SoC, we can read the revision information from the timer > >>+ * block. The detection logic is extracted from similar code on the > >>+ * Allwinner vendor tree, as this is undocumented on the user manual > >>+ */ > >>+ > >>+#define TIMER_SOC_REV_REG 0x13c > >>+#define TIMER_SOC_REV_CLEAR(val) ((val) & ~(0x3 << 6)) > >>+#define TIMER_SOC_REV_GET(val) (((val) >> 6) & 0x3) > >>+ > >>+static const struct of_device_id sun4i_timer_compatible[] __initconst = { > >>+ { .compatible = "allwinner,sun4i-a10-timer", }, > >>+ {}, > >>+}; > >>+ > >>+static int __init sun4i_get_soc_revision(void) > >>+{ > >>+ struct device_node *np; > >>+ void __iomem *base; > >>+ u32 val; > >>+ int ret; > >>+ > >>+ /* Find the timer node */ > >>+ np = of_find_matching_node(NULL, sun4i_timer_compatible); > >>+ if (!np) > >>+ return -ENODEV; > >>+ > >>+ /* Temporarily map it for reading */ > >>+ base = of_iomap(np, 0); > >>+ if (!base) { > >>+ of_node_put(np); > >>+ return -ENOMEM; > >>+ } > >>+ > >>+ /* Clear the SoC revision bits and rewrite the register */ > >>+ val = readl(base + TIMER_SOC_REV_REG); > >>+ val = TIMER_SOC_REV_CLEAR(val); > >>+ writel(val, base + TIMER_SOC_REV_REG); > >>+ > >>+ /* Now read it again and see what shows up */ > >>+ val = readl(base + TIMER_SOC_REV_REG); > >>+ val = TIMER_SOC_REV_GET(val); > >>+ > >>+ switch (val) { > >>+ case 0: /* revision A */ > >>+ ret = 'A'; > >>+ case 3: /* revision B */ > >>+ ret = 'B'; > >>+ default: /* revision C */ > >>+ ret = 'C'; > > > >What's programmed in there in the case of the rev C? > > Something that's not a 3 or a 0 - I don't have any more specifics > than what the Allwinner code does :( > > https://github.com/linux-sunxi/linux-sunxi/blob/lichee-dev/arch/arm/mach-sun4i/core.c#L418 It would be interesting to see if there's a value programmed. > I just noticed that I forgot the breaks while reworking this from > return to a variable, so everything is rev C right now - I'll fix > that. > > >>+ } > >>+ > >>+ iounmap(base); > >>+ of_node_put(np); > >>+ > >>+ return ret; > >>+} > >>+ > >>+/* > >>+ * On the sun5i SoCs (A10S, A13), we can read the revision information > >>+ * from the first bits in the Security ID. The detection logic is > >>+ * extracted from similar code on the Allwinner vendor tree, as this > >>+ * is undocumented on the user manual. > >>+ */ > >>+ > >>+static const struct of_device_id sun5i_sid_compatible[] __initconst = { > >>+ { .compatible = "allwinner,sun4i-a10-sid", }, > >>+ {}, > >>+}; > >>+ > >>+static int __init sun5i_get_soc_revision(void) > >>+{ > >>+ struct device_node *np; > >>+ void __iomem *sid; > >>+ u32 val; > >>+ int ret; > >>+ > >>+ /* Find the SID node */ > >>+ np = of_find_matching_node(NULL, sun5i_sid_compatible); > >>+ if (!np) > >>+ return -ENODEV; > >>+ > >>+ /* Temporarily map it for reading */ > >>+ sid = of_iomap(np, 0); > >>+ if (!sid) { > >>+ of_node_put(np); > >>+ return -ENOMEM; > >>+ } > >>+ > >>+ /* Read and extract the chip revision from the SID */ > >>+ val = readl(sid); > >>+ val = (val >> 8) & 0xffffff; > >>+ > >>+ switch (val) { > >>+ case 0: /* A10S/A13 rev A */ > >>+ case 0x162541: /* A10S/A13 rev A */ > >>+ case 0x162565: /* A13 rev A */ > > > >Some defines would be nice. > > Ok > > >Also, isn't the SID supposed to identify any SoC, not just the sun5i? > > The Allwinner code I've seen uses the SS to detect the SoC (which > luckily we don't need, as we have DT) and then reads the timer for > sun4i or the SID for sun5i. SS being security system or system controller ? :) > > >>+ ret = 'A'; > >>+ break; > >>+ case 0x162542: /* A10S/A13 rev B */ > >>+ ret = 'B'; > >>+ break; > >>+ default: /* Unknown chip revision */ > >>+ ret = -ENODATA; > >>+ } > >>+ > >>+ iounmap(sid); > >>+ of_node_put(np); > >>+ > >>+ return ret; > >>+} > >>+ > >>+int __init sunxi_soc_revision(void) > >>+{ > >>+ static int revision = -ENODEV; > >>+ > >>+ /* Try to query the hardware just once */ > >>+ if (!IS_ERR_VALUE(revision)) > >>+ return revision; > >>+ > >>+ if (of_machine_is_compatible("allwinner,sun4i-a10")) { > >>+ revision = sun4i_get_soc_revision(); > >>+ } else if (of_machine_is_compatible("allwinner,sun5i-a10s") || > >>+ of_machine_is_compatible("allwinner,sun5i-a13")) { > >>+ revision = sun5i_get_soc_revision(); > >>+ } > >>+ > >>+ return revision; > >>+} > >>+ > >>+/* Matches for the sunxi SoCs we know of */ > >>+static const struct of_device_id soc_matches[] __initconst = { > >>+ { .compatible = "allwinner,sun4i-a10", .data = "A10 (sun4i)" }, > >>+ { .compatible = "allwinner,sun5i-a13", .data = "A13 (sun5i)" }, > >>+ { .compatible = "allwinner,sun5i-a10s", .data = "A10S (sun5i)" }, > >>+ { .compatible = "allwinner,sun6i-a31", .data = "A31 (sun6i)" }, > >>+ { .compatible = "allwinner,sun7i-a20", .data = "A20 (sun7i)" }, > >>+ { .compatible = "allwinner,sun8i-a23", .data = "A23 (sun8i)" }, > >>+ { /* sentinel */ }, > >>+}; > > > >Hmmm, no, either auto-detect it, or don't, but that's useless. We can > >already have the same information from the DT directly. > > I don't quite get what you're trying to say here. This is here to > a) get a fancy string saying what the SoC is (AFAIK that's not > directly available on the DT) > b) only run in the listed SoCs I'm saying that the point of this code is to auto-detect the SoC ID/rev, so taking that info from the DT brings nothing compared to using of_machine_is_compatible. > >>+static int __init sunxi_register_soc_device(void) > >>+{ > >>+ struct soc_device_attribute *soc_dev_attr; > >>+ struct soc_device *soc_dev; > >>+ const struct of_device_id *match; > >>+ struct device_node *root; > >>+ int revision; > >>+ > >>+ /* Only run on sunxi SoCs that we know of */ > >>+ root = of_find_node_by_path("/"); > >>+ match = of_match_node(soc_matches, root); > >>+ if (!match) > >>+ goto exit; > > See here > > >>+ > >>+ soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL); > >>+ if (!soc_dev_attr) > >>+ goto exit; > >>+ > >>+ /* Read the machine name if available */ > >>+ of_property_read_string(root, "model", &soc_dev_attr->machine); > >>+ > >>+ soc_dev_attr->family = kstrdup("Allwinner A Series", GFP_KERNEL); > > > >I think the family should be sun*i > > Here is a list of the ones currently in use for family > > "Cirrus Logic EP93xx" > "Freescale i.MX" > "Integrator" > "Marvell" > "Freescale MXS Family" > "Tegra" > "Xilinx Zynq" > > There does not seem to be a real consensus on what these mean. > There's a binding document on > Documentation/ABI/testing/sysfs-devices-soc but it's not really > detailed. > > > > >>+ soc_dev_attr->soc_id = kstrdup(match->data, GFP_KERNEL); > > > >And soc_id would be just the name of the SoC. > > According to the binding this is a serial number. Lee, could you > help us decide what these fields should look like? > > I'm now inclining to have eg. family=sun5i, machine=A13, keep > revision the same and drop soc_id. Ah, my guess would have been that the soc_id would be the AW16XX. > > >>+ > >>+ /* Revision may not always be available */ > >>+ revision = sunxi_soc_revision(); > >>+ if (IS_ERR_VALUE(revision)) > >>+ soc_dev_attr->revision = kstrdup("Unknown", GFP_KERNEL); > >>+ else > >>+ soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%c", revision); > >>+ > >>+ soc_dev = soc_device_register(soc_dev_attr); > >>+ if (IS_ERR(soc_dev)) > >>+ goto free_struct; > >>+ > >>+ /* > >>+ * Print an informational line mentioning the hardware details. > >>+ * It may come in handy during bug reports, as some early SoC > >>+ * revisions have hardware quirks and do not get much testing. > >>+ */ > >>+ pr_info("SoC bus registered, running %s %s, revision %s\n", > >>+ soc_dev_attr->family, soc_dev_attr->soc_id, > >>+ soc_dev_attr->revision); > > > >Maybe a pr_fmt would be nice here. > > With something like "Allwinner SoC bus"? "sunxi" would be enough. > > >>+ > >>+ return 0; > >>+ > >>+free_struct: > >>+ kfree(soc_dev_attr->family); > >>+ kfree(soc_dev_attr->soc_id); > >>+ kfree(soc_dev_attr->revision); > >>+ kfree(soc_dev_attr); > >>+exit: > >>+ of_node_put(root); > >>+ > >>+ return 0; > >>+} > >>+postcore_initcall(sunxi_register_soc_device) > > > >I'm kind of reluctant to this. That would mean that it will run for > >every SoC. > > The overhead should be negligible - on non-sunxi, it will only check > if the machine is compatible by looking at the root node and bail > out as it's not the case. > > >Can't you tie it to the system controller, and have it probed as > >usual? > > I considered tying this to init_machine, right before the DT starts > to get probed. That would mean actually writing init_machine hooks > calling into this for some of our machines, and getting us further > from the generic DT machine nirvana though. As the quirk bits do not > directly depend on this, we can probably get away with making this a > normal > driver compatible with all our SoCs. Let me know which way do you > prefer, and I'll be happy to oblige. Isn't one of the point of the soc_device_register to be that the registered soc_device would be the parent device of all the SoC devices? (There's probably too many devices and soc words in that sentence though). That would make the registration of the soc_device before the call to of_platform_populate mandatory. If not, we can see it two-folds then: - First, setup the quirks, possibly caching any relevant values - Then, use a regular driver to register the soc_device with the infos you gathered from the cache Maxime
diff --git a/arch/arm/mach-sunxi/Kconfig b/arch/arm/mach-sunxi/Kconfig index 6434e3b..4a199df 100644 --- a/arch/arm/mach-sunxi/Kconfig +++ b/arch/arm/mach-sunxi/Kconfig @@ -5,6 +5,7 @@ menuconfig ARCH_SUNXI select GENERIC_IRQ_CHIP select PINCTRL select PINCTRL_SUNXI + select SOC_BUS select SUN4I_TIMER if ARCH_SUNXI diff --git a/arch/arm/mach-sunxi/Makefile b/arch/arm/mach-sunxi/Makefile index 27b168f..589239b 100644 --- a/arch/arm/mach-sunxi/Makefile +++ b/arch/arm/mach-sunxi/Makefile @@ -1,2 +1,2 @@ -obj-$(CONFIG_ARCH_SUNXI) += sunxi.o +obj-$(CONFIG_ARCH_SUNXI) += sunxi.o sunxi-soc-id.o obj-$(CONFIG_SMP) += platsmp.o diff --git a/arch/arm/mach-sunxi/sunxi-soc-id.c b/arch/arm/mach-sunxi/sunxi-soc-id.c new file mode 100644 index 0000000..c7eff1c --- /dev/null +++ b/arch/arm/mach-sunxi/sunxi-soc-id.c @@ -0,0 +1,226 @@ +/* + * SoC revision detection for sunxi SoCs + * + * Copyright 2014 Emilio López + * + * Emilio López <emilio@elopez.com.ar> + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include <linux/err.h> +#include <linux/io.h> +#include <linux/of.h> +#include <linux/of_address.h> +#include <linux/slab.h> +#include <linux/string.h> +#include <linux/sys_soc.h> + +#include "sunxi-soc-id.h" + +/* + * On the A10 SoC, we can read the revision information from the timer + * block. The detection logic is extracted from similar code on the + * Allwinner vendor tree, as this is undocumented on the user manual + */ + +#define TIMER_SOC_REV_REG 0x13c +#define TIMER_SOC_REV_CLEAR(val) ((val) & ~(0x3 << 6)) +#define TIMER_SOC_REV_GET(val) (((val) >> 6) & 0x3) + +static const struct of_device_id sun4i_timer_compatible[] __initconst = { + { .compatible = "allwinner,sun4i-a10-timer", }, + {}, +}; + +static int __init sun4i_get_soc_revision(void) +{ + struct device_node *np; + void __iomem *base; + u32 val; + int ret; + + /* Find the timer node */ + np = of_find_matching_node(NULL, sun4i_timer_compatible); + if (!np) + return -ENODEV; + + /* Temporarily map it for reading */ + base = of_iomap(np, 0); + if (!base) { + of_node_put(np); + return -ENOMEM; + } + + /* Clear the SoC revision bits and rewrite the register */ + val = readl(base + TIMER_SOC_REV_REG); + val = TIMER_SOC_REV_CLEAR(val); + writel(val, base + TIMER_SOC_REV_REG); + + /* Now read it again and see what shows up */ + val = readl(base + TIMER_SOC_REV_REG); + val = TIMER_SOC_REV_GET(val); + + switch (val) { + case 0: /* revision A */ + ret = 'A'; + case 3: /* revision B */ + ret = 'B'; + default: /* revision C */ + ret = 'C'; + } + + iounmap(base); + of_node_put(np); + + return ret; +} + +/* + * On the sun5i SoCs (A10S, A13), we can read the revision information + * from the first bits in the Security ID. The detection logic is + * extracted from similar code on the Allwinner vendor tree, as this + * is undocumented on the user manual. + */ + +static const struct of_device_id sun5i_sid_compatible[] __initconst = { + { .compatible = "allwinner,sun4i-a10-sid", }, + {}, +}; + +static int __init sun5i_get_soc_revision(void) +{ + struct device_node *np; + void __iomem *sid; + u32 val; + int ret; + + /* Find the SID node */ + np = of_find_matching_node(NULL, sun5i_sid_compatible); + if (!np) + return -ENODEV; + + /* Temporarily map it for reading */ + sid = of_iomap(np, 0); + if (!sid) { + of_node_put(np); + return -ENOMEM; + } + + /* Read and extract the chip revision from the SID */ + val = readl(sid); + val = (val >> 8) & 0xffffff; + + switch (val) { + case 0: /* A10S/A13 rev A */ + case 0x162541: /* A10S/A13 rev A */ + case 0x162565: /* A13 rev A */ + ret = 'A'; + break; + case 0x162542: /* A10S/A13 rev B */ + ret = 'B'; + break; + default: /* Unknown chip revision */ + ret = -ENODATA; + } + + iounmap(sid); + of_node_put(np); + + return ret; +} + +int __init sunxi_soc_revision(void) +{ + static int revision = -ENODEV; + + /* Try to query the hardware just once */ + if (!IS_ERR_VALUE(revision)) + return revision; + + if (of_machine_is_compatible("allwinner,sun4i-a10")) { + revision = sun4i_get_soc_revision(); + } else if (of_machine_is_compatible("allwinner,sun5i-a10s") || + of_machine_is_compatible("allwinner,sun5i-a13")) { + revision = sun5i_get_soc_revision(); + } + + return revision; +} + +/* Matches for the sunxi SoCs we know of */ +static const struct of_device_id soc_matches[] __initconst = { + { .compatible = "allwinner,sun4i-a10", .data = "A10 (sun4i)" }, + { .compatible = "allwinner,sun5i-a13", .data = "A13 (sun5i)" }, + { .compatible = "allwinner,sun5i-a10s", .data = "A10S (sun5i)" }, + { .compatible = "allwinner,sun6i-a31", .data = "A31 (sun6i)" }, + { .compatible = "allwinner,sun7i-a20", .data = "A20 (sun7i)" }, + { .compatible = "allwinner,sun8i-a23", .data = "A23 (sun8i)" }, + { /* sentinel */ }, +}; + +static int __init sunxi_register_soc_device(void) +{ + struct soc_device_attribute *soc_dev_attr; + struct soc_device *soc_dev; + const struct of_device_id *match; + struct device_node *root; + int revision; + + /* Only run on sunxi SoCs that we know of */ + root = of_find_node_by_path("/"); + match = of_match_node(soc_matches, root); + if (!match) + goto exit; + + soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL); + if (!soc_dev_attr) + goto exit; + + /* Read the machine name if available */ + of_property_read_string(root, "model", &soc_dev_attr->machine); + + soc_dev_attr->family = kstrdup("Allwinner A Series", GFP_KERNEL); + soc_dev_attr->soc_id = kstrdup(match->data, GFP_KERNEL); + + /* Revision may not always be available */ + revision = sunxi_soc_revision(); + if (IS_ERR_VALUE(revision)) + soc_dev_attr->revision = kstrdup("Unknown", GFP_KERNEL); + else + soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%c", revision); + + soc_dev = soc_device_register(soc_dev_attr); + if (IS_ERR(soc_dev)) + goto free_struct; + + /* + * Print an informational line mentioning the hardware details. + * It may come in handy during bug reports, as some early SoC + * revisions have hardware quirks and do not get much testing. + */ + pr_info("SoC bus registered, running %s %s, revision %s\n", + soc_dev_attr->family, soc_dev_attr->soc_id, + soc_dev_attr->revision); + + return 0; + +free_struct: + kfree(soc_dev_attr->family); + kfree(soc_dev_attr->soc_id); + kfree(soc_dev_attr->revision); + kfree(soc_dev_attr); +exit: + of_node_put(root); + + return 0; +} +postcore_initcall(sunxi_register_soc_device) diff --git a/arch/arm/mach-sunxi/sunxi-soc-id.h b/arch/arm/mach-sunxi/sunxi-soc-id.h new file mode 100644 index 0000000..d49c245 --- /dev/null +++ b/arch/arm/mach-sunxi/sunxi-soc-id.h @@ -0,0 +1,6 @@ +#ifndef _SUNXI_SOC_ID_H +#define _SUNXI_SOC_ID_H + +int __init sunxi_soc_revision(void); + +#endif /* _SUNXI_SOC_ID_H */
This commit adds SoC bus support on the sunxi platform, and exposes information such as the hardware revision to userspace and other kernel clients during init. A message with this information is also printed to the kernel log to ease future bug triaging. Signed-off-by: Emilio López <emilio@elopez.com.ar> --- arch/arm/mach-sunxi/Kconfig | 1 + arch/arm/mach-sunxi/Makefile | 2 +- arch/arm/mach-sunxi/sunxi-soc-id.c | 226 +++++++++++++++++++++++++++++++++++++ arch/arm/mach-sunxi/sunxi-soc-id.h | 6 + 4 files changed, 234 insertions(+), 1 deletion(-) create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.c create mode 100644 arch/arm/mach-sunxi/sunxi-soc-id.h