psci.c 1.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859
  1. /*
  2. * (C) Copyright 2015, Siemens AG
  3. * Author: Jan Kiszka <jan.kiszka@siemens.com>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <asm/io.h>
  9. #include <asm/psci.h>
  10. #include <asm/arch/flow.h>
  11. #include <asm/arch/powergate.h>
  12. #include <asm/arch-tegra/ap.h>
  13. #include <asm/arch-tegra/pmc.h>
  14. static void park_cpu(void)
  15. {
  16. while (1)
  17. asm volatile("wfi");
  18. }
  19. /**
  20. * Initialize power management for application processors
  21. */
  22. void psci_board_init(void)
  23. {
  24. struct flow_ctlr *flow = (struct flow_ctlr *)NV_PA_FLOW_BASE;
  25. writel((u32)park_cpu, EXCEP_VECTOR_CPU_RESET_VECTOR);
  26. /*
  27. * The naturally expected order of putting these CPUs under Flow
  28. * Controller regime would be
  29. * - configure the Flow Controller
  30. * - power up the CPUs
  31. * - wait for the CPUs to hit wfi and be powered down again
  32. *
  33. * However, this doesn't work in practice. We rather need to power them
  34. * up first and park them in wfi. While they are waiting there, we can
  35. * indeed program the Flow Controller to powergate them on wfi, which
  36. * will then happen immediately as they are already in that state.
  37. */
  38. tegra_powergate_power_on(TEGRA_POWERGATE_CPU1);
  39. tegra_powergate_power_on(TEGRA_POWERGATE_CPU2);
  40. tegra_powergate_power_on(TEGRA_POWERGATE_CPU3);
  41. writel((2 << CSR_WAIT_WFI_SHIFT) | CSR_ENABLE, &flow->cpu1_csr);
  42. writel((4 << CSR_WAIT_WFI_SHIFT) | CSR_ENABLE, &flow->cpu2_csr);
  43. writel((8 << CSR_WAIT_WFI_SHIFT) | CSR_ENABLE, &flow->cpu3_csr);
  44. writel(EVENT_MODE_STOP, &flow->halt_cpu1_events);
  45. writel(EVENT_MODE_STOP, &flow->halt_cpu2_events);
  46. writel(EVENT_MODE_STOP, &flow->halt_cpu3_events);
  47. while (!(readl(&flow->cpu1_csr) & CSR_PWR_OFF_STS) ||
  48. !(readl(&flow->cpu2_csr) & CSR_PWR_OFF_STS) ||
  49. !(readl(&flow->cpu3_csr) & CSR_PWR_OFF_STS))
  50. /* wait */;
  51. }