blob: 4cea0b9ae3e4093f78a1d10e7005a40a41e1f353 [file] [log] [blame]
Soby Mathew0d786072016-03-24 16:56:29 +00001/*
Boyan Karatotev02d9d832024-11-06 16:23:07 +00002 * Copyright (c) 2016-2025, Arm Limited and Contributors. All rights reserved.
Varun Wadekar0a176e32020-02-13 13:07:12 -08003 * Copyright (c) 2020, NVIDIA Corporation. All rights reserved.
Soby Mathew0d786072016-03-24 16:56:29 +00004 *
dp-armfa3cf0b2017-05-03 09:38:09 +01005 * SPDX-License-Identifier: BSD-3-Clause
Soby Mathew0d786072016-03-24 16:56:29 +00006 */
7
Soby Mathew0d786072016-03-24 16:56:29 +00008#include <assert.h>
Antonio Nino Diaze0f90632018-12-14 00:18:21 +00009
10#include <arch.h>
11#include <lib/pmf/pmf.h>
12#include <lib/psci/psci.h>
Varun Wadekar0a176e32020-02-13 13:07:12 -080013#include <lib/utils_def.h>
Antonio Nino Diaze0f90632018-12-14 00:18:21 +000014#include <plat/common/platform.h>
Soby Mathew0d786072016-03-24 16:56:29 +000015
dp-arm66abfbe2017-01-31 13:01:04 +000016#if ENABLE_PSCI_STAT && ENABLE_PMF
17#pragma weak plat_psci_stat_accounting_start
18#pragma weak plat_psci_stat_accounting_stop
19#pragma weak plat_psci_stat_get_residency
20
Soby Mathew8336f682017-10-16 15:19:31 +010021/* Maximum time-stamp value read from architectural counters */
Julius Werner8e0ef0f2019-07-09 14:02:43 -070022#ifdef __aarch64__
Soby Mathew8336f682017-10-16 15:19:31 +010023#define MAX_TS UINT64_MAX
Julius Werner8e0ef0f2019-07-09 14:02:43 -070024#else
25#define MAX_TS UINT32_MAX
Soby Mathew8336f682017-10-16 15:19:31 +010026#endif
27
dp-arm66abfbe2017-01-31 13:01:04 +000028/* Following are used as ID's to capture time-stamp */
29#define PSCI_STAT_ID_ENTER_LOW_PWR 0
30#define PSCI_STAT_ID_EXIT_LOW_PWR 1
31#define PSCI_STAT_TOTAL_IDS 2
32
Boyan Karatotev02d9d832024-11-06 16:23:07 +000033#if HW_ASSISTED_COHERENCY
34#define CACHE_MAINTENANCE_ATTR PMF_NO_CACHE_MAINT
35#else
36#define CACHE_MAINTENANCE_ATTR PMF_CACHE_MAINT
37#endif
38
Madhukar Pappireddya091d042020-01-06 14:42:30 -060039PMF_DECLARE_CAPTURE_TIMESTAMP(psci_svc)
40PMF_DECLARE_GET_TIMESTAMP(psci_svc)
dp-arm66abfbe2017-01-31 13:01:04 +000041PMF_REGISTER_SERVICE(psci_svc, PMF_PSCI_STAT_SVC_ID, PSCI_STAT_TOTAL_IDS,
42 PMF_STORE_ENABLE)
43
44/*
45 * This function calculates the stats residency in microseconds,
46 * taking in account the wrap around condition.
47 */
48static u_register_t calc_stat_residency(unsigned long long pwrupts,
49 unsigned long long pwrdnts)
50{
51 /* The divisor to use to convert raw timestamp into microseconds. */
52 u_register_t residency_div;
53 u_register_t res;
54
55 /*
56 * Calculate divisor so that it can be directly used to
57 * convert time-stamp into microseconds.
58 */
59 residency_div = read_cntfrq_el0() / MHZ_TICKS_PER_SEC;
Antonio Nino Diaz6f3ccc52018-07-20 09:17:26 +010060 assert(residency_div > 0U);
dp-arm66abfbe2017-01-31 13:01:04 +000061
62 if (pwrupts < pwrdnts)
Soby Mathew8336f682017-10-16 15:19:31 +010063 res = MAX_TS - pwrdnts + pwrupts;
dp-arm66abfbe2017-01-31 13:01:04 +000064 else
65 res = pwrupts - pwrdnts;
66
67 return res / residency_div;
68}
69
70/*
71 * Capture timestamp before entering a low power state.
dp-arm66abfbe2017-01-31 13:01:04 +000072 * Cache maintenance may be needed when reading these timestamps.
73 */
74void plat_psci_stat_accounting_start(
75 __unused const psci_power_state_t *state_info)
76{
Antonio Nino Diazfec756f2018-07-18 16:24:16 +010077 assert(state_info != NULL);
dp-arm66abfbe2017-01-31 13:01:04 +000078 PMF_CAPTURE_TIMESTAMP(psci_svc, PSCI_STAT_ID_ENTER_LOW_PWR,
Boyan Karatotev02d9d832024-11-06 16:23:07 +000079 CACHE_MAINTENANCE_ATTR);
dp-arm66abfbe2017-01-31 13:01:04 +000080}
81
82/*
83 * Capture timestamp after exiting a low power state.
dp-arm66abfbe2017-01-31 13:01:04 +000084 * Cache maintenance may be needed when reading these timestamps.
85 */
86void plat_psci_stat_accounting_stop(
87 __unused const psci_power_state_t *state_info)
88{
Antonio Nino Diazfec756f2018-07-18 16:24:16 +010089 assert(state_info != NULL);
dp-arm66abfbe2017-01-31 13:01:04 +000090 PMF_CAPTURE_TIMESTAMP(psci_svc, PSCI_STAT_ID_EXIT_LOW_PWR,
Boyan Karatotev02d9d832024-11-06 16:23:07 +000091 CACHE_MAINTENANCE_ATTR);
dp-arm66abfbe2017-01-31 13:01:04 +000092}
93
94/*
95 * Calculate the residency for the given level and power state
96 * information.
97 */
98u_register_t plat_psci_stat_get_residency(unsigned int lvl,
99 const psci_power_state_t *state_info,
Deepika Bhavnani4287c0c2019-12-13 10:23:18 -0600100 unsigned int last_cpu_idx)
dp-arm66abfbe2017-01-31 13:01:04 +0000101{
dp-arm66abfbe2017-01-31 13:01:04 +0000102 unsigned long long pwrup_ts = 0, pwrdn_ts = 0;
103 unsigned int pmf_flags;
104
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100105 assert((lvl >= PSCI_CPU_PWR_LVL) && (lvl <= PLAT_MAX_PWR_LVL));
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100106 assert(last_cpu_idx <= PLATFORM_CORE_COUNT);
dp-arm66abfbe2017-01-31 13:01:04 +0000107
108 if (lvl == PSCI_CPU_PWR_LVL)
Deepika Bhavnani4287c0c2019-12-13 10:23:18 -0600109 assert(last_cpu_idx == plat_my_core_pos());
dp-arm66abfbe2017-01-31 13:01:04 +0000110
Boyan Karatotev02d9d832024-11-06 16:23:07 +0000111#if HW_ASSISTED_COHERENCY
112 /* HW coherency allows for the capture and access to happen with caches
113 * ON. So these timestamps don't need cache maintenance */
114 pmf_flags = PMF_NO_CACHE_MAINT;
115#else
dp-arm66abfbe2017-01-31 13:01:04 +0000116 /*
117 * If power down is requested, then timestamp capture will
118 * be with caches OFF. Hence we have to do cache maintenance
119 * when reading the timestamp.
120 */
Boyan Karatotev02d9d832024-11-06 16:23:07 +0000121 plat_local_state_t state;
122 assert(state_info != NULL);
dp-arm66abfbe2017-01-31 13:01:04 +0000123 state = state_info->pwr_domain_state[PSCI_CPU_PWR_LVL];
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100124 if (is_local_state_off(state) != 0) {
dp-arm66abfbe2017-01-31 13:01:04 +0000125 pmf_flags = PMF_CACHE_MAINT;
126 } else {
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100127 assert(is_local_state_retn(state) == 1);
dp-arm66abfbe2017-01-31 13:01:04 +0000128 pmf_flags = PMF_NO_CACHE_MAINT;
129 }
Boyan Karatotev02d9d832024-11-06 16:23:07 +0000130#endif
dp-arm66abfbe2017-01-31 13:01:04 +0000131
132 PMF_GET_TIMESTAMP_BY_INDEX(psci_svc,
133 PSCI_STAT_ID_ENTER_LOW_PWR,
134 last_cpu_idx,
135 pmf_flags,
136 pwrdn_ts);
137
138 PMF_GET_TIMESTAMP_BY_INDEX(psci_svc,
139 PSCI_STAT_ID_EXIT_LOW_PWR,
140 plat_my_core_pos(),
141 pmf_flags,
142 pwrup_ts);
143
144 return calc_stat_residency(pwrup_ts, pwrdn_ts);
145}
146#endif /* ENABLE_PSCI_STAT && ENABLE_PMF */
147
Soby Mathew0d786072016-03-24 16:56:29 +0000148/*
149 * The PSCI generic code uses this API to let the platform participate in state
150 * coordination during a power management operation. It compares the platform
151 * specific local power states requested by each cpu for a given power domain
152 * and returns the coordinated target power state that the domain should
153 * enter. A platform assigns a number to a local power state. This default
154 * implementation assumes that the platform assigns these numbers in order of
155 * increasing depth of the power state i.e. for two power states X & Y, if X < Y
156 * then X represents a shallower power state than Y. As a result, the
157 * coordinated target local power state for a power domain will be the minimum
158 * of the requested local power states.
159 */
160plat_local_state_t plat_get_target_pwr_state(unsigned int lvl,
161 const plat_local_state_t *states,
162 unsigned int ncpu)
163{
164 plat_local_state_t target = PLAT_MAX_OFF_STATE, temp;
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100165 const plat_local_state_t *st = states;
166 unsigned int n = ncpu;
Soby Mathew0d786072016-03-24 16:56:29 +0000167
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100168 assert(ncpu > 0U);
Soby Mathew0d786072016-03-24 16:56:29 +0000169
170 do {
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100171 temp = *st;
172 st++;
Soby Mathew0d786072016-03-24 16:56:29 +0000173 if (temp < target)
174 target = temp;
Antonio Nino Diazfec756f2018-07-18 16:24:16 +0100175 n--;
176 } while (n > 0U);
Soby Mathew0d786072016-03-24 16:56:29 +0000177
178 return target;
179}