[PATCH v6 4/7] arm64: add PSCI CPU_SUSPEND based cpu_suspend support

Ashwin Chaugule ashwin.chaugule at linaro.org
Wed Jul 23 09:15:15 PDT 2014


On 21 July 2014 12:06, Lorenzo Pieralisi <lorenzo.pieralisi at arm.com> wrote:
> This patch implements the cpu_suspend cpu operations method through
> the PSCI CPU SUSPEND API. The PSCI implementation translates the idle state
> index passed by the cpu_suspend core call into a valid PSCI state according to
> the PSCI states initialized at boot through the cpu_init_idle() CPU
> operations hook.
>
> Entry point is set to cpu_resume physical address, that represents the
> default kernel execution address following a CPU reset; for standby
> states the entry point address is useless and will therefore be ignored
> by the PSCI implementation.
>
> Signed-off-by: Lorenzo Pieralisi <lorenzo.pieralisi at arm.com>
> ---
>  arch/arm64/kernel/psci.c | 89 ++++++++++++++++++++++++++++++++++++++++++++++++
>  1 file changed, 89 insertions(+)
>
> diff --git a/arch/arm64/kernel/psci.c b/arch/arm64/kernel/psci.c
> index a623c44..bbdf41d 100644
> --- a/arch/arm64/kernel/psci.c
> +++ b/arch/arm64/kernel/psci.c
> @@ -21,6 +21,7 @@
>  #include <linux/reboot.h>
>  #include <linux/pm.h>
>  #include <linux/delay.h>
> +#include <linux/slab.h>
>  #include <uapi/linux/psci.h>
>
>  #include <asm/compiler.h>
> @@ -28,6 +29,7 @@
>  #include <asm/errno.h>
>  #include <asm/psci.h>
>  #include <asm/smp_plat.h>
> +#include <asm/suspend.h>
>  #include <asm/system_misc.h>
>
>  #define PSCI_POWER_STATE_TYPE_STANDBY          0
> @@ -65,6 +67,8 @@ enum psci_function {
>         PSCI_FN_MAX,
>  };
>
> +static DEFINE_PER_CPU_READ_MOSTLY(struct psci_power_state *, psci_power_state);
> +
>  static u32 psci_function_id[PSCI_FN_MAX];
>
>  static int psci_to_linux_errno(int errno)
> @@ -93,6 +97,18 @@ static u32 psci_power_state_pack(struct psci_power_state state)
>                  & PSCI_0_2_POWER_STATE_AFFL_MASK);
>  }
>
> +static void psci_power_state_unpack(u32 power_state,
> +                                   struct psci_power_state *state)
> +{
> +       state->id = (power_state & PSCI_0_2_POWER_STATE_ID_MASK) >>
> +                       PSCI_0_2_POWER_STATE_ID_SHIFT;
> +       state->type = (power_state & PSCI_0_2_POWER_STATE_TYPE_MASK) >>
> +                       PSCI_0_2_POWER_STATE_TYPE_SHIFT;
> +       state->affinity_level =
> +                       (power_state & PSCI_0_2_POWER_STATE_AFFL_MASK) >>
> +                       PSCI_0_2_POWER_STATE_AFFL_SHIFT;
> +}
> +
>  /*
>   * The following two functions are invoked via the invoke_psci_fn pointer
>   * and will not be inlined, allowing us to piggyback on the AAPCS.
> @@ -199,6 +215,61 @@ static int psci_migrate_info_type(void)
>         return err;
>  }
>
> +static int cpu_psci_cpu_init_idle(struct device_node *cpu_node,
> +                                 unsigned int cpu)
> +{
> +       int i, ret, count = 0;
> +       struct psci_power_state *psci_states;
> +       struct device_node *state_node;
> +
> +       /*
> +        * If the PSCI cpu_suspend function hook has not been initialized
> +        * idle states must not be enabled, so bail out
> +        */
> +       if (!psci_ops.cpu_suspend)
> +               return -EOPNOTSUPP;
> +
> +       /* Count idle states */
> +       while ((state_node = of_parse_phandle(cpu_node, "cpu-idle-states",
> +                                             count)))
> +               count++;
> +
> +       if (!count)
> +               return -ENODEV;
> +
> +       psci_states = kcalloc(count, sizeof(*psci_states), GFP_KERNEL);
> +       if (!psci_states) {
> +               pr_warn("idle state psci_state allocation failed\n");
> +               return -ENOMEM;
> +       }
> +
> +       for (i = 0; i < count; i++) {
> +               u32 psci_power_state;
> +
> +               state_node = of_parse_phandle(cpu_node, "cpu-idle-states", i);
> +
> +               ret = of_property_read_u32(state_node,
> +                                          "arm,psci-suspend-param",
> +                                          &psci_power_state);
> +               if (ret) {
> +                       pr_warn(" * %s missing arm,psci-suspend-param property\n",
> +                               state_node->full_name);
> +                       goto free_mem;
> +               }
> +
> +               pr_debug("psci-power-state %#x index %d\n", psci_power_state,
> +                                                           i);
> +               psci_power_state_unpack(psci_power_state, &psci_states[i]);
> +       }
> +       /* Idle states parsed correctly, initialize per-cpu pointer */
> +       per_cpu(psci_power_state, cpu) = psci_states;
> +       return 0;
> +
> +free_mem:
> +       kfree(psci_states);
> +       return ret;
> +}
> +
>  static int get_set_conduit_method(struct device_node *np)
>  {
>         const char *method;
> @@ -436,8 +507,23 @@ static int cpu_psci_cpu_kill(unsigned int cpu)
>  #endif
>  #endif
>
> +static int __maybe_unused cpu_psci_cpu_suspend(unsigned long index)
> +{
> +       struct psci_power_state *state = __get_cpu_var(psci_power_state);
> +       /*
> +        * idle state index 0 corresponds to wfi, should never be called
> +        * from the cpu_suspend operations
> +        */
> +       if (WARN_ON_ONCE(!index))
> +               return -EINVAL;
> +
> +       return psci_ops.cpu_suspend(state[index - 1],
> +                                   virt_to_phys(cpu_resume));
> +}
> +
>  const struct cpu_operations cpu_psci_ops = {
>         .name           = "psci",
> +       .cpu_init_idle  = cpu_psci_cpu_init_idle,
>  #ifdef CONFIG_SMP
>         .cpu_init       = cpu_psci_cpu_init,
>         .cpu_prepare    = cpu_psci_cpu_prepare,
> @@ -448,5 +534,8 @@ const struct cpu_operations cpu_psci_ops = {
>         .cpu_kill       = cpu_psci_cpu_kill,
>  #endif
>  #endif
> +#ifdef CONFIG_ARM64_CPU_SUSPEND
> +       .cpu_suspend    = cpu_psci_cpu_suspend,
> +#endif
>  };
>
> --
> 1.9.1

Reviewed-by: Ashwin Chaugule <ashwin.chaugule at linaro.org>

>
>
>
> _______________________________________________
> linux-arm-kernel mailing list
> linux-arm-kernel at lists.infradead.org
> http://lists.infradead.org/mailman/listinfo/linux-arm-kernel



More information about the linux-arm-kernel mailing list