On Nov 1, 2013, at 5:08 PM, Stephen Boyd wrote:

> From: Rohit Vaswani <rvasw...@codeaurora.org>
> 
> This makes it easy to add SMP support for new devices by keying
> on a device node for the release sequence. We add the
> enable-method property for the cpus property to specify that we
> want to use the mmio release sequence (which is going to look for
> a device node to map some Scorpion specific power and control
> registers). While at it, add the 8660 cpus bindings to make SMP
> work.
> 
> Signed-off-by: Rohit Vaswani <rvasw...@codeaurora.org>
> [sboyd: Port to CPU_METHOD_OF_DECLARE, replace if/else with match
> table method]
> Signed-off-by: Stephen Boyd <sb...@codeaurora.org>
> ---
> arch/arm/boot/dts/qcom-msm8660-surf.dts |  24 ++++++++
> arch/arm/mach-msm/board-dt.c            |  14 -----
> arch/arm/mach-msm/common.h              |   1 -
> arch/arm/mach-msm/platsmp.c             | 103 ++++++++++++++++++++++++--------
> 4 files changed, 103 insertions(+), 39 deletions(-)

This feels like a few different patches munged together.

> diff --git a/arch/arm/boot/dts/qcom-msm8660-surf.dts 
> b/arch/arm/boot/dts/qcom-msm8660-surf.dts
> index 386d428..ac5ea76 100644
> --- a/arch/arm/boot/dts/qcom-msm8660-surf.dts
> +++ b/arch/arm/boot/dts/qcom-msm8660-surf.dts
> @@ -7,6 +7,23 @@
>       compatible = "qcom,msm8660-surf", "qcom,msm8660";
>       interrupt-parent = <&intc>;
> 
> +     cpus {
> +             #address-cells = <1>;
> +             #size-cells = <0>;
> +             compatible = "qcom,scorpion";
> +             enable-method = "qcom,mmio";
> +
> +             cpu@0 {
> +                     device_type = "cpu";
> +                     reg = <0>;
> +             };
> +
> +             cpu@1 {
> +                     device_type = "cpu";
> +                     reg = <1>;
> +             };
> +     };
> +
>       intc: interrupt-controller@2080000 {
>               compatible = "qcom,msm-8660-qgic";
>               interrupt-controller;
> @@ -37,6 +54,13 @@
>               #interrupt-cells = <2>;
>       };
> 
> +     clock-controller@900000 {
> +             compatible = "qcom,gcc-8660";
> +             reg = <0x00900000 0x4000>;
> +             #clock-cells = <1>;
> +             #reset-cells = <1>;
> +     };
> +
>       serial@19c40000 {
>               compatible = "qcom,msm-uartdm-v1.3", "qcom,msm-uartdm";
>               reg = <0x19c40000 0x1000>,
> diff --git a/arch/arm/mach-msm/board-dt.c b/arch/arm/mach-msm/board-dt.c
> index 1f11d93..1e3af2b 100644
> --- a/arch/arm/mach-msm/board-dt.c
> +++ b/arch/arm/mach-msm/board-dt.c
> @@ -11,31 +11,17 @@
>  */
> 
> #include <linux/init.h>
> -#include <linux/of.h>
> -#include <linux/of_platform.h>
> 
> #include <asm/mach/arch.h>
> -#include <asm/mach/map.h>
> -
> -#include "common.h"
> 
> static const char * const msm_dt_match[] __initconst = {
>       "qcom,msm8660-fluid",
>       "qcom,msm8660-surf",
>       "qcom,msm8960-cdp",
> -     NULL
> -};
> -
> -static const char * const apq8074_dt_match[] __initconst = {
>       "qcom,apq8074-dragonboard",
>       NULL
> };
> 
> DT_MACHINE_START(MSM_DT, "Qualcomm MSM (Flattened Device Tree)")
> -     .smp = smp_ops(msm_smp_ops),
>       .dt_compat = msm_dt_match,
> MACHINE_END
> -
> -DT_MACHINE_START(APQ_DT, "Qualcomm MSM (Flattened Device Tree)")
> -     .dt_compat = apq8074_dt_match,
> -MACHINE_END
> diff --git a/arch/arm/mach-msm/common.h b/arch/arm/mach-msm/common.h
> index 33c7725..1513f2c 100644
> --- a/arch/arm/mach-msm/common.h
> +++ b/arch/arm/mach-msm/common.h
> @@ -23,7 +23,6 @@ extern void msm_map_qsd8x50_io(void);
> extern void __iomem *__msm_ioremap_caller(phys_addr_t phys_addr, size_t size,
>                                         unsigned int mtype, void *caller);
> 
> -extern struct smp_operations msm_smp_ops;
> extern void msm_cpu_die(unsigned int cpu);
> 
> struct msm_mmc_platform_data;
> diff --git a/arch/arm/mach-msm/platsmp.c b/arch/arm/mach-msm/platsmp.c
> index 5b481db..369d8f7 100644
> --- a/arch/arm/mach-msm/platsmp.c
> +++ b/arch/arm/mach-msm/platsmp.c
> @@ -13,6 +13,8 @@
> #include <linux/delay.h>
> #include <linux/device.h>
> #include <linux/jiffies.h>
> +#include <linux/of.h>
> +#include <linux/of_address.h>
> #include <linux/smp.h>
> #include <linux/io.h>
> 
> @@ -24,9 +26,9 @@
> #include "scm-boot.h"
> #include "common.h"
> 
> -#define VDD_SC1_ARRAY_CLAMP_GFS_CTL 0x15A0
> -#define SCSS_CPU1CORE_RESET 0xD80
> -#define SCSS_DBG_STATUS_CORE_PWRDUP 0xE64
> +#define VDD_SC1_ARRAY_CLAMP_GFS_CTL  0x35a0
> +#define SCSS_CPU1CORE_RESET          0x2d80
> +#define SCSS_DBG_STATUS_CORE_PWRDUP  0x2e64
> 
> extern void secondary_startup(void);
> 
> @@ -47,34 +49,62 @@ static void msm_secondary_init(unsigned int cpu)
>       spin_unlock(&boot_lock);
> }
> 
> -static void prepare_cold_cpu(unsigned int cpu)
> +static int scss_release_secondary(struct device_node *node, unsigned int cpu)
> +{
> +     void __iomem *base;
> +
> +     base = of_iomap(node, 0);
> +     if (!base)
> +             return -ENOMEM;
> +
> +     writel_relaxed(0, base + VDD_SC1_ARRAY_CLAMP_GFS_CTL);
> +     writel_relaxed(0, base + SCSS_CPU1CORE_RESET);
> +     writel_relaxed(3, base + SCSS_DBG_STATUS_CORE_PWRDUP);
> +     mb();
> +     iounmap(base);
> +
> +     return 0;
> +}
> +
> +static DEFINE_PER_CPU(int, cold_boot_done);
> +
> +static int boot_cold_cpu(unsigned int cpu)
> {
>       int ret;
> -     ret = scm_set_boot_addr(virt_to_phys(secondary_startup),
> -                             SCM_FLAG_COLDBOOT_CPU1);
> -     if (ret == 0) {
> -             void __iomem *sc1_base_ptr;
> -             sc1_base_ptr = ioremap_nocache(0x00902000, SZ_4K*2);
> -             if (sc1_base_ptr) {
> -                     writel(0, sc1_base_ptr + VDD_SC1_ARRAY_CLAMP_GFS_CTL);
> -                     writel(0, sc1_base_ptr + SCSS_CPU1CORE_RESET);
> -                     writel(3, sc1_base_ptr + SCSS_DBG_STATUS_CORE_PWRDUP);
> -                     iounmap(sc1_base_ptr);
> -             }
> -     } else
> -             printk(KERN_DEBUG "Failed to set secondary core boot "
> -                               "address\n");
> +     struct device_node *node;
> +     const struct of_device_id *match;
> +     const int (*func)(struct device_node *, int);
> +#define M(c, f) { .compatible = c, .data = f }
> +     static const struct of_device_id match_table[] = {
> +             M("qcom,gcc-8660", scss_release_secondary),
> +     };
> +#undef M

Uughh..

Can't we just do a of_find_compatible_node(), however I think this will change 
based on my other comments.

> +
> +     if (per_cpu(cold_boot_done, cpu))
> +             return 0;
> +
> +     node = of_find_matching_node_and_match(NULL, match_table, &match);
> +     if (!node) {
> +             pr_err("%s: can't find subsystem node\n", __func__);
> +             return -ENXIO;
> +     }
> +
> +     func = match->data;
> +     ret = func(node, cpu);
> +     if (!ret)
> +             per_cpu(cold_boot_done, cpu) = true;
> +     of_node_put(node);
> +
> +     return ret;
> }
> 
> static int msm_boot_secondary(unsigned int cpu, struct task_struct *idle)
> {
> -     static int cold_boot_done;
> +     int ret;
> 
> -     /* Only need to bring cpu out of reset this way once */
> -     if (cold_boot_done == false) {
> -             prepare_cold_cpu(cpu);
> -             cold_boot_done = true;
> -     }
> +     ret = boot_cold_cpu(cpu);
> +     if (ret)
> +             return ret;
> 
>       /*
>        * set synchronisation state between this boot processor
> @@ -120,6 +150,30 @@ static void __init msm_smp_init_cpus(void)
> 
> static void __init msm_smp_prepare_cpus(unsigned int max_cpus)
> {
> +     int cpu, map;
> +     unsigned int flags = 0;
> +     static const int cold_boot_flags[] = {
> +             0,
> +             SCM_FLAG_COLDBOOT_CPU1,
> +     };
> +
> +     for_each_present_cpu(cpu) {
> +             map = cpu_logical_map(cpu);
> +             if (WARN_ON(map >= ARRAY_SIZE(cold_boot_flags))) {
> +                     set_cpu_present(cpu, false);
> +                     continue;
> +             }
> +             flags |= cold_boot_flags[map];
> +     }
> +
> +     if (scm_set_boot_addr(virt_to_phys(secondary_startup), flags)) {
> +             for_each_present_cpu(cpu) {
> +                     if (cpu == smp_processor_id())
> +                             continue;
> +                     set_cpu_present(cpu, false);
> +             }
> +             pr_warn("Failed to set CPU boot address, disabling SMP\n");
> +     }
> }
> 
> struct smp_operations msm_smp_ops __initdata = {
> @@ -131,3 +185,4 @@ struct smp_operations msm_smp_ops __initdata = {
>       .cpu_die                = msm_cpu_die,
> #endif
> };
> +CPU_METHOD_OF_DECLARE(msm_smp, "qcom,mmio", &msm_smp_ops);
> -- 
> The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum,
> hosted by The Linux Foundation
> 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-arm-msm" in
> the body of a message to majord...@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

-- 
Employee of Qualcomm Innovation Center, Inc.
Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum, hosted by 
The Linux Foundation

--
To unsubscribe from this list: send the line "unsubscribe linux-arm-msm" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to