mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-16 01:54:00 +00:00
825f4e0271
A quite large set of SoC updates this cycle. In no particular order: - Multi-cluster power management for Samsung Exynos, adding support for big.LITTLE CPU switching on EXYNOS5 - SMP support for Marvell Armada 375 and 38x - SMP rework on Allwinner A31 - Xilinx Zynq support for SOC_BUS, big endian - Marvell orion5x platform cleanup, modernizing the implementation and moving to DT. - _Finally_ moving Samsung Exynos over to support MULTIPLATFORM, so that their platform can be enabled in the same kernel binary as most of the other v7 platforms in the tree. \o/ The work isn't quite complete, there's some driver fixes still needed, but the basics now work. New SoC support added: - Freescale i.MX6SX - LSI Axxia AXM55xx SoCs - Samsung EXYNOS 3250, 5260, 5410, 5420 and 5800 - STi STIH407 Plus a large set of various smaller updates for different platforms. I'm probably missing some important one here. -----BEGIN PGP SIGNATURE----- Version: GnuPG v1.4.14 (GNU/Linux) iQIcBAABAgAGBQJTjOKWAAoJEIwa5zzehBx36aEP/2vTD7x9FC59FACNHJ8iO7aw 0ebTgBBjI1Np6X18O+M7URbxV5TaBgwpUm/NDN86p03MpQ2eOXr8r47qVxe/HhZs AdlTvzgE6QwxcVL/HeCKKUEN3BPH74+TZgFl9I5aSzNjpR39xETeK1aWP/ZiAl/q /lGRZAQ59+c7Ung00Hg0g2YDxH9WFpK50Nj90ROnyjKSFkhIYngXYVpZB3maOypq Pgib/U8IraKZ52oGJw3yinSoORr7FdcUdAGWGTz/lQdNL/jYDfQ6GkRW2oblWXdt 3Xvj9UW6NmkbMICucMvFuuW1nXAgutZuTp9w7mBxsiUlYepxPv/DXM6yiI1WGlEb BeVOmOreNeN2nT6avv/uUhk3Osq63Jn9x8cz5y+7/lgWQwllh3/c+G01RotvgJEQ vpQq5ps9fMxIAMaNP6N/YqMJI1IOrBj0iXxaZEDw3VYM/k4lSvtb3VXP9c/rqApu U4i6hpSIGzrraU4NrjndYPndcLeNOVZbByETQKosZXuCo6G1sb7FstNSkzI9vSo8 O/pujIVUfYyBW82GzZGDw+aa7DWA29FPeUQ3p+sj5MSCg051xXT8h6QwqMo2K/zY 5ATs/qo6w7zH/Ou9rtHTRynCIb0GQJThDSlWtuXFedUF9quEltS+TDz/2o+dWtGJ yBFGKDRuBB20D36w9xqg =6LYI -----END PGP SIGNATURE----- Merge tag 'soc-for-3.16' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc into next Pull part one of ARM SoC updates from Olof Johansson: "A quite large set of SoC updates this cycle. In no particular order: - Multi-cluster power management for Samsung Exynos, adding support for big.LITTLE CPU switching on EXYNOS5 - SMP support for Marvell Armada 375 and 38x - SMP rework on Allwinner A31 - Xilinx Zynq support for SOC_BUS, big endian - Marvell orion5x platform cleanup, modernizing the implementation and moving to DT. - _Finally_ moving Samsung Exynos over to support MULTIPLATFORM, so that their platform can be enabled in the same kernel binary as most of the other v7 platforms in the tree. \o/ The work isn't quite complete, there's some driver fixes still needed, but the basics now work. New SoC support added: - Freescale i.MX6SX - LSI Axxia AXM55xx SoCs - Samsung EXYNOS 3250, 5260, 5410, 5420 and 5800 - STi STIH407 plus a large set of various smaller updates for different platforms. I'm probably missing some important one here" * tag 'soc-for-3.16' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc: (281 commits) ARM: exynos: don't run exynos4 l2x0 setup on other platforms ARM: exynos: Fix "allmodconfig" build errors in mcpm and hotplug ARM: EXYNOS: mcpm rename the power_down_finish ARM: EXYNOS: Enable mcpm for dual-cluster exynos5800 SoC ARM: EXYNOS: Enable multi-platform build support ARM: EXYNOS: Consolidate Kconfig entries ARM: EXYNOS: Add support for EXYNOS5410 SoC ARM: EXYNOS: Support secondary CPU boot of Exynos3250 ARM: EXYNOS: Add Exynos3250 SoC ID ARM: EXYNOS: Add 5800 SoC support ARM: EXYNOS: initial board support for exynos5260 SoC clk: exynos5410: register clocks using common clock framework ARM: debug: qcom: add UART addresses to Kconfig help for APQ8084 ARM: sunxi: allow building without reset controller Documentation: devicetree: arm: sort enable-method entries ARM: rockchip: convert smp bringup to CPU_METHOD_OF_DECLARE clk: exynos5250: Add missing sysmmu clocks for DISP and ISP blocks ARM: dts: axxia: Add reset controller power: reset: Add Axxia system reset driver ARM: axxia: Adding defconfig for AXM55xx ...
162 lines
3.7 KiB
C
162 lines
3.7 KiB
C
/*
|
|
* ID and revision information for mvebu SoCs
|
|
*
|
|
* Copyright (C) 2014 Marvell
|
|
*
|
|
* Gregory CLEMENT <gregory.clement@free-electrons.com>
|
|
*
|
|
* This file is licensed under the terms of the GNU General Public
|
|
* License version 2. This program is licensed "as is" without any
|
|
* warranty of any kind, whether express or implied.
|
|
*
|
|
* All the mvebu SoCs have information related to their variant and
|
|
* revision that can be read from the PCI control register. This is
|
|
* done before the PCI initialization to avoid any conflict. Once the
|
|
* ID and revision are retrieved, the mapping is freed.
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "mvebu-soc-id: " fmt
|
|
|
|
#include <linux/clk.h>
|
|
#include <linux/init.h>
|
|
#include <linux/io.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_address.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/sys_soc.h>
|
|
#include "mvebu-soc-id.h"
|
|
|
|
#define PCIE_DEV_ID_OFF 0x0
|
|
#define PCIE_DEV_REV_OFF 0x8
|
|
|
|
#define SOC_ID_MASK 0xFFFF0000
|
|
#define SOC_REV_MASK 0xFF
|
|
|
|
static u32 soc_dev_id;
|
|
static u32 soc_rev;
|
|
static bool is_id_valid;
|
|
|
|
static const struct of_device_id mvebu_pcie_of_match_table[] = {
|
|
{ .compatible = "marvell,armada-xp-pcie", },
|
|
{ .compatible = "marvell,armada-370-pcie", },
|
|
{ .compatible = "marvell,kirkwood-pcie" },
|
|
{},
|
|
};
|
|
|
|
int mvebu_get_soc_id(u32 *dev, u32 *rev)
|
|
{
|
|
if (is_id_valid) {
|
|
*dev = soc_dev_id;
|
|
*rev = soc_rev;
|
|
return 0;
|
|
} else
|
|
return -1;
|
|
}
|
|
|
|
static int __init mvebu_soc_id_init(void)
|
|
{
|
|
struct device_node *np;
|
|
int ret = 0;
|
|
void __iomem *pci_base;
|
|
struct clk *clk;
|
|
struct device_node *child;
|
|
|
|
np = of_find_matching_node(NULL, mvebu_pcie_of_match_table);
|
|
if (!np)
|
|
return ret;
|
|
|
|
/*
|
|
* ID and revision are available from any port, so we
|
|
* just pick the first one
|
|
*/
|
|
child = of_get_next_child(np, NULL);
|
|
if (child == NULL) {
|
|
pr_err("cannot get pci node\n");
|
|
ret = -ENOMEM;
|
|
goto clk_err;
|
|
}
|
|
|
|
clk = of_clk_get_by_name(child, NULL);
|
|
if (IS_ERR(clk)) {
|
|
pr_err("cannot get clock\n");
|
|
ret = -ENOMEM;
|
|
goto clk_err;
|
|
}
|
|
|
|
ret = clk_prepare_enable(clk);
|
|
if (ret) {
|
|
pr_err("cannot enable clock\n");
|
|
goto clk_err;
|
|
}
|
|
|
|
pci_base = of_iomap(child, 0);
|
|
if (pci_base == NULL) {
|
|
pr_err("cannot map registers\n");
|
|
ret = -ENOMEM;
|
|
goto res_ioremap;
|
|
}
|
|
|
|
/* SoC ID */
|
|
soc_dev_id = readl(pci_base + PCIE_DEV_ID_OFF) >> 16;
|
|
|
|
/* SoC revision */
|
|
soc_rev = readl(pci_base + PCIE_DEV_REV_OFF) & SOC_REV_MASK;
|
|
|
|
is_id_valid = true;
|
|
|
|
pr_info("MVEBU SoC ID=0x%X, Rev=0x%X\n", soc_dev_id, soc_rev);
|
|
|
|
iounmap(pci_base);
|
|
|
|
res_ioremap:
|
|
/*
|
|
* If the PCIe unit is actually enabled and we have PCI
|
|
* support in the kernel, we intentionally do not release the
|
|
* reference to the clock. We want to keep it running since
|
|
* the bootloader does some PCIe link configuration that the
|
|
* kernel is for now unable to do, and gating the clock would
|
|
* make us loose this precious configuration.
|
|
*/
|
|
if (!of_device_is_available(child) || !IS_ENABLED(CONFIG_PCI_MVEBU)) {
|
|
clk_disable_unprepare(clk);
|
|
clk_put(clk);
|
|
}
|
|
|
|
clk_err:
|
|
of_node_put(child);
|
|
of_node_put(np);
|
|
|
|
return ret;
|
|
}
|
|
early_initcall(mvebu_soc_id_init);
|
|
|
|
static int __init mvebu_soc_device(void)
|
|
{
|
|
struct soc_device_attribute *soc_dev_attr;
|
|
struct soc_device *soc_dev;
|
|
|
|
/* Also protects against running on non-mvebu systems */
|
|
if (!is_id_valid)
|
|
return 0;
|
|
|
|
soc_dev_attr = kzalloc(sizeof(*soc_dev_attr), GFP_KERNEL);
|
|
if (!soc_dev_attr)
|
|
return -ENOMEM;
|
|
|
|
soc_dev_attr->family = kasprintf(GFP_KERNEL, "Marvell");
|
|
soc_dev_attr->revision = kasprintf(GFP_KERNEL, "%X", soc_rev);
|
|
soc_dev_attr->soc_id = kasprintf(GFP_KERNEL, "%X", soc_dev_id);
|
|
|
|
soc_dev = soc_device_register(soc_dev_attr);
|
|
if (IS_ERR(soc_dev)) {
|
|
kfree(soc_dev_attr->family);
|
|
kfree(soc_dev_attr->revision);
|
|
kfree(soc_dev_attr->soc_id);
|
|
kfree(soc_dev_attr);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
postcore_initcall(mvebu_soc_device);
|