blob: 86459585eafec97ab35f284f4690e6ad321b97aa [file] [log] [blame]
Mikael Olsson7da66192021-02-12 17:30:22 +01001/*
Joshua Pimm6bc80672022-10-19 15:46:27 +01002 * Copyright (c) 2021-2023, Arm Limited. All rights reserved.
Mikael Olsson7da66192021-02-12 17:30:22 +01003 *
4 * SPDX-License-Identifier: BSD-3-Clause
5 */
6
7#include <stdint.h>
8#include <stdbool.h>
9
10#include <common/debug.h>
11#include <common/runtime_svc.h>
12#include <drivers/arm/ethosn.h>
13#include <drivers/delay_timer.h>
14#include <lib/mmio.h>
Mikael Olsson3288b462022-08-15 17:12:58 +020015#include <lib/utils_def.h>
Mikael Olsson7da66192021-02-12 17:30:22 +010016#include <plat/arm/common/fconf_ethosn_getter.h>
17
Laurent Carlier5205df22021-09-16 15:10:35 +010018/*
Mikael Olsson3288b462022-08-15 17:12:58 +020019 * Number of Arm(R) Ethos(TM)-N NPU (NPU) devices available
Laurent Carlier5205df22021-09-16 15:10:35 +010020 */
Mikael Olsson3288b462022-08-15 17:12:58 +020021#define ETHOSN_NUM_DEVICES \
22 FCONF_GET_PROPERTY(hw_config, ethosn_config, num_devices)
Mikael Olsson7da66192021-02-12 17:30:22 +010023
Mikael Olsson3288b462022-08-15 17:12:58 +020024#define ETHOSN_GET_DEVICE(dev_idx) \
25 FCONF_GET_PROPERTY(hw_config, ethosn_device, dev_idx)
Mikael Olsson7da66192021-02-12 17:30:22 +010026
27/* NPU core sec registry address */
28#define ETHOSN_CORE_SEC_REG(core_addr, reg_offset) \
29 (core_addr + reg_offset)
30
31/* Reset timeout in us */
32#define ETHOSN_RESET_TIMEOUT_US U(10 * 1000 * 1000)
33#define ETHOSN_RESET_WAIT_US U(1)
34
35#define SEC_DEL_REG U(0x0004)
36#define SEC_DEL_VAL U(0x81C)
37#define SEC_DEL_EXCC_MASK U(0x20)
38
39#define SEC_SECCTLR_REG U(0x0010)
40#define SEC_SECCTLR_VAL U(0x3)
41
Mikael Olsson7da66192021-02-12 17:30:22 +010042#define SEC_DEL_ADDR_EXT_REG U(0x201C)
43#define SEC_DEL_ADDR_EXT_VAL U(0x15)
44
45#define SEC_SYSCTRL0_REG U(0x0018)
Mikael Olsson47675f22022-11-04 15:01:02 +010046#define SEC_SYSCTRL0_SLEEPING U(1U << 4)
Mikael Olsson7da66192021-02-12 17:30:22 +010047#define SEC_SYSCTRL0_SOFT_RESET U(3U << 29)
48#define SEC_SYSCTRL0_HARD_RESET U(1U << 31)
49
Mikael Olsson3288b462022-08-15 17:12:58 +020050#define SEC_MMUSID_REG_BASE U(0x3008)
51#define SEC_MMUSID_OFFSET U(0x1000)
52
53static bool ethosn_get_device_and_core(uintptr_t core_addr,
54 const struct ethosn_device_t **dev_match,
55 const struct ethosn_core_t **core_match)
Laurent Carlier5205df22021-09-16 15:10:35 +010056{
Mikael Olsson3288b462022-08-15 17:12:58 +020057 uint32_t dev_idx;
58 uint32_t core_idx;
59
60 for (dev_idx = 0U; dev_idx < ETHOSN_NUM_DEVICES; ++dev_idx) {
61 const struct ethosn_device_t *dev = ETHOSN_GET_DEVICE(dev_idx);
62
63 for (core_idx = 0U; core_idx < dev->num_cores; ++core_idx) {
64 const struct ethosn_core_t *core = &(dev->cores[core_idx]);
65
66 if (core->addr == core_addr) {
67 *dev_match = dev;
68 *core_match = core;
69 return true;
70 }
Laurent Carlier5205df22021-09-16 15:10:35 +010071 }
72 }
73
Mikael Olsson3288b462022-08-15 17:12:58 +020074 WARN("ETHOSN: Unknown core address given to SMC call.\n");
Laurent Carlier5205df22021-09-16 15:10:35 +010075 return false;
76}
77
Mikael Olsson3288b462022-08-15 17:12:58 +020078static void ethosn_configure_smmu_streams(const struct ethosn_device_t *device,
79 const struct ethosn_core_t *core,
80 uint32_t asset_alloc_idx)
81{
82 const struct ethosn_main_allocator_t *main_alloc =
83 &(core->main_allocator);
84 const struct ethosn_asset_allocator_t *asset_alloc =
85 &(device->asset_allocators[asset_alloc_idx]);
86 const uint32_t streams[9] = {
87 main_alloc->firmware.stream_id,
88 main_alloc->working_data.stream_id,
89 asset_alloc->command_stream.stream_id,
90 0U, /* Not used*/
91 main_alloc->firmware.stream_id,
92 asset_alloc->weight_data.stream_id,
93 asset_alloc->buffer_data.stream_id,
94 asset_alloc->intermediate_data.stream_id,
95 asset_alloc->buffer_data.stream_id
96 };
97 size_t i;
98
99 for (i = 0U; i < ARRAY_SIZE(streams); ++i) {
100 const uintptr_t reg_addr = SEC_MMUSID_REG_BASE +
101 (SEC_MMUSID_OFFSET * i);
102 mmio_write_32(ETHOSN_CORE_SEC_REG(core->addr, reg_addr),
103 streams[i]);
104 }
105}
106
Mikael Olsson7da66192021-02-12 17:30:22 +0100107static void ethosn_delegate_to_ns(uintptr_t core_addr)
108{
109 mmio_setbits_32(ETHOSN_CORE_SEC_REG(core_addr, SEC_SECCTLR_REG),
110 SEC_SECCTLR_VAL);
111
112 mmio_setbits_32(ETHOSN_CORE_SEC_REG(core_addr, SEC_DEL_REG),
113 SEC_DEL_VAL);
114
Mikael Olsson7da66192021-02-12 17:30:22 +0100115 mmio_setbits_32(ETHOSN_CORE_SEC_REG(core_addr, SEC_DEL_ADDR_EXT_REG),
116 SEC_DEL_ADDR_EXT_VAL);
117}
118
Laurent Carlier5205df22021-09-16 15:10:35 +0100119static int ethosn_is_sec(uintptr_t core_addr)
Mikael Olsson7da66192021-02-12 17:30:22 +0100120{
Laurent Carlier5205df22021-09-16 15:10:35 +0100121 if ((mmio_read_32(ETHOSN_CORE_SEC_REG(core_addr, SEC_DEL_REG))
Mikael Olsson7da66192021-02-12 17:30:22 +0100122 & SEC_DEL_EXCC_MASK) != 0U) {
123 return 0;
124 }
125
126 return 1;
127}
128
Mikael Olsson47675f22022-11-04 15:01:02 +0100129static int ethosn_core_is_sleeping(uintptr_t core_addr)
130{
131 const uintptr_t sysctrl0_reg =
132 ETHOSN_CORE_SEC_REG(core_addr, SEC_SYSCTRL0_REG);
133 const uint32_t sleeping_mask = SEC_SYSCTRL0_SLEEPING;
134
135 return ((mmio_read_32(sysctrl0_reg) & sleeping_mask) == sleeping_mask);
136}
137
Mikael Olsson7da66192021-02-12 17:30:22 +0100138static bool ethosn_reset(uintptr_t core_addr, int hard_reset)
139{
140 unsigned int timeout;
141 const uintptr_t sysctrl0_reg =
142 ETHOSN_CORE_SEC_REG(core_addr, SEC_SYSCTRL0_REG);
143 const uint32_t reset_val = (hard_reset != 0) ? SEC_SYSCTRL0_HARD_RESET
144 : SEC_SYSCTRL0_SOFT_RESET;
145
146 mmio_write_32(sysctrl0_reg, reset_val);
147
148 /* Wait for reset to complete */
149 for (timeout = 0U; timeout < ETHOSN_RESET_TIMEOUT_US;
150 timeout += ETHOSN_RESET_WAIT_US) {
151
152 if ((mmio_read_32(sysctrl0_reg) & reset_val) == 0U) {
153 break;
154 }
155
156 udelay(ETHOSN_RESET_WAIT_US);
157 }
158
159 return timeout < ETHOSN_RESET_TIMEOUT_US;
160}
161
162uintptr_t ethosn_smc_handler(uint32_t smc_fid,
Laurent Carlier5205df22021-09-16 15:10:35 +0100163 u_register_t core_addr,
Mikael Olsson3288b462022-08-15 17:12:58 +0200164 u_register_t asset_alloc_idx,
Joshua Pimm6bc80672022-10-19 15:46:27 +0100165 u_register_t reset_type,
Mikael Olsson7da66192021-02-12 17:30:22 +0100166 u_register_t x4,
167 void *cookie,
168 void *handle,
169 u_register_t flags)
170{
Mikael Olsson7da66192021-02-12 17:30:22 +0100171 int hard_reset = 0;
Mikael Olsson3288b462022-08-15 17:12:58 +0200172 const struct ethosn_device_t *device = NULL;
173 const struct ethosn_core_t *core = NULL;
Laurent Carlier5205df22021-09-16 15:10:35 +0100174 const uint32_t fid = smc_fid & FUNCID_NUM_MASK;
Mikael Olsson7da66192021-02-12 17:30:22 +0100175
176 /* Only SiP fast calls are expected */
177 if ((GET_SMC_TYPE(smc_fid) != SMC_TYPE_FAST) ||
178 (GET_SMC_OEN(smc_fid) != OEN_SIP_START)) {
179 SMC_RET1(handle, SMC_UNK);
180 }
181
182 /* Truncate parameters to 32-bits for SMC32 */
183 if (GET_SMC_CC(smc_fid) == SMC_32) {
Laurent Carlier5205df22021-09-16 15:10:35 +0100184 core_addr &= 0xFFFFFFFF;
Mikael Olsson3288b462022-08-15 17:12:58 +0200185 asset_alloc_idx &= 0xFFFFFFFF;
Joshua Pimm6bc80672022-10-19 15:46:27 +0100186 reset_type &= 0xFFFFFFFF;
Mikael Olsson7da66192021-02-12 17:30:22 +0100187 x4 &= 0xFFFFFFFF;
188 }
189
Mikael Olsson47675f22022-11-04 15:01:02 +0100190 if (!is_ethosn_fid(smc_fid) || (fid > ETHOSN_FNUM_IS_SLEEPING)) {
Mikael Olsson3288b462022-08-15 17:12:58 +0200191 WARN("ETHOSN: Unknown SMC call: 0x%x\n", smc_fid);
Mikael Olsson7da66192021-02-12 17:30:22 +0100192 SMC_RET1(handle, SMC_UNK);
193 }
194
Laurent Carlier5205df22021-09-16 15:10:35 +0100195 /* Commands that do not require a valid core address */
196 switch (fid) {
Mikael Olsson7da66192021-02-12 17:30:22 +0100197 case ETHOSN_FNUM_VERSION:
198 SMC_RET2(handle, ETHOSN_VERSION_MAJOR, ETHOSN_VERSION_MINOR);
Laurent Carlier5205df22021-09-16 15:10:35 +0100199 }
200
Mikael Olsson3288b462022-08-15 17:12:58 +0200201 if (!ethosn_get_device_and_core(core_addr, &device, &core)) {
Laurent Carlier5205df22021-09-16 15:10:35 +0100202 SMC_RET1(handle, ETHOSN_UNKNOWN_CORE_ADDRESS);
203 }
204
Mikael Olsson3288b462022-08-15 17:12:58 +0200205 /* Commands that require a valid core address */
Laurent Carlier5205df22021-09-16 15:10:35 +0100206 switch (fid) {
Mikael Olsson7da66192021-02-12 17:30:22 +0100207 case ETHOSN_FNUM_IS_SEC:
Mikael Olsson3288b462022-08-15 17:12:58 +0200208 SMC_RET1(handle, ethosn_is_sec(core->addr));
Mikael Olsson47675f22022-11-04 15:01:02 +0100209 case ETHOSN_FNUM_IS_SLEEPING:
210 SMC_RET1(handle, ethosn_core_is_sleeping(core->addr));
Mikael Olsson3288b462022-08-15 17:12:58 +0200211 }
212
213 if (!device->has_reserved_memory &&
214 asset_alloc_idx >= device->num_allocators) {
215 WARN("ETHOSN: Unknown asset allocator index given to SMC call.\n");
216 SMC_RET1(handle, ETHOSN_UNKNOWN_ALLOCATOR_IDX);
217 }
218
Joshua Pimm6bc80672022-10-19 15:46:27 +0100219 if (reset_type > ETHOSN_RESET_TYPE_HALT) {
220 WARN("ETHOSN: Invalid reset type given to SMC call.\n");
221 SMC_RET1(handle, ETHOSN_INVALID_PARAMETER);
222 }
223
224 /*
225 * Commands that require a valid device, reset type,
226 * core and asset allocator
227 */
Mikael Olsson3288b462022-08-15 17:12:58 +0200228 switch (fid) {
Mikael Olsson7da66192021-02-12 17:30:22 +0100229 case ETHOSN_FNUM_HARD_RESET:
230 hard_reset = 1;
231 /* Fallthrough */
232 case ETHOSN_FNUM_SOFT_RESET:
Mikael Olsson3288b462022-08-15 17:12:58 +0200233 if (!ethosn_reset(core->addr, hard_reset)) {
Mikael Olsson7da66192021-02-12 17:30:22 +0100234 SMC_RET1(handle, ETHOSN_FAILURE);
235 }
Mikael Olsson3288b462022-08-15 17:12:58 +0200236
Joshua Pimm6bc80672022-10-19 15:46:27 +0100237 if (reset_type == ETHOSN_RESET_TYPE_FULL) {
238 if (!device->has_reserved_memory) {
239 ethosn_configure_smmu_streams(device, core,
240 asset_alloc_idx);
241 }
Mikael Olsson3288b462022-08-15 17:12:58 +0200242
Joshua Pimm6bc80672022-10-19 15:46:27 +0100243 ethosn_delegate_to_ns(core->addr);
244 }
Mikael Olsson7da66192021-02-12 17:30:22 +0100245 SMC_RET1(handle, ETHOSN_SUCCESS);
246 default:
Mikael Olsson3288b462022-08-15 17:12:58 +0200247 WARN("ETHOSN: Unimplemented SMC call: 0x%x\n", fid);
Mikael Olsson7da66192021-02-12 17:30:22 +0100248 SMC_RET1(handle, SMC_UNK);
249 }
250}