/* * SPDX-FileCopyrightText: Copyright 2021-2022, 2024 Arm Limited and/or its * affiliates * SPDX-License-Identifier: Apache-2.0 */ #include "zephyr/sys_clock.h" #include #include #include #include #include #include #include #include LOG_MODULE_REGISTER(ethos_u, CONFIG_ARM_ETHOS_U_LOG_LEVEL); #define DT_DRV_COMPAT arm_ethos_u /******************************************************************************* * Re-implementation/Overrides __((weak)) symbol functions from ethosu_driver.c * To handle mutex and semaphores *******************************************************************************/ void *ethosu_mutex_create(void) { struct k_mutex *mutex; mutex = k_malloc(sizeof(*mutex)); if (mutex == NULL) { LOG_ERR("Failed allocate mutex"); return NULL; } k_mutex_init(mutex); return (void *)mutex; } int ethosu_mutex_lock(void *mutex) { int status; status = k_mutex_lock((struct k_mutex *)mutex, K_FOREVER); if (status != 0) { LOG_ERR("Failed to lock mutex with error - %d", status); return -1; } return 0; } int ethosu_mutex_unlock(void *mutex) { k_mutex_unlock((struct k_mutex *)mutex); return 0; } void *ethosu_semaphore_create(void) { struct k_sem *sem; sem = k_malloc(sizeof(*sem)); if (sem == NULL) { LOG_ERR("Failed to allocate semaphore"); return NULL; } k_sem_init(sem, 0, 100); return (void *)sem; } int ethosu_semaphore_take(void *sem, uint64_t timeout) { int status; status = k_sem_take((struct k_sem *)sem, (timeout == ETHOSU_SEMAPHORE_WAIT_FOREVER) ? K_FOREVER : Z_TIMEOUT_TICKS(timeout)); if (status != 0) { /* The Ethos-U driver expects the semaphore implementation to never fail except for * when a timeout occurs, and the current ethosu_semaphore_take implementation makes * no distinction, in terms of return codes, between a timeout and other semaphore * take failures. Also, note that a timeout is virtually indistinguishable from * other failures if the driver logging is disabled. Handling errors other than a * timeout is therefore not covered here and is deferred to the application * developer if necessary. */ if (status != -EAGAIN) { LOG_ERR("Failed to take semaphore with error - %d", status); } return -1; } return 0; } int ethosu_semaphore_give(void *sem) { k_sem_give((struct k_sem *)sem); return 0; } struct ethosu_dts_info { void *base_addr; bool secure_enable; bool privilege_enable; void (*irq_config)(void); }; struct ethosu_data { struct ethosu_driver drv; }; void ethosu_zephyr_irq_handler(const struct device *dev) { struct ethosu_data *data = dev->data; struct ethosu_driver *drv = &data->drv; ethosu_irq_handler(drv); } static int ethosu_zephyr_init(const struct device *dev) { const struct ethosu_dts_info *config = dev->config; struct ethosu_data *data = dev->data; struct ethosu_driver *drv = &data->drv; struct ethosu_driver_version version; LOG_DBG("Ethos-U DTS info. base_address=0x%p, secure_enable=%u, privilege_enable=%u", config->base_addr, config->secure_enable, config->privilege_enable); ethosu_get_driver_version(&version); LOG_DBG("Version. major=%u, minor=%u, patch=%u", version.major, version.minor, version.patch); if (ethosu_init(drv, config->base_addr, NULL, 0, config->secure_enable, config->privilege_enable)) { LOG_ERR("Failed to initialize NPU with ethosu_init()."); return -EINVAL; } config->irq_config(); return 0; } #define ETHOSU_DEVICE_INIT(n) \ static struct ethosu_data ethosu_data_##n; \ \ static void ethosu_zephyr_irq_config_##n(void) \ { \ IRQ_CONNECT(DT_INST_IRQN(n), DT_INST_IRQ(n, priority), ethosu_zephyr_irq_handler, \ DEVICE_DT_INST_GET(n), 0); \ irq_enable(DT_INST_IRQN(n)); \ } \ \ static const struct ethosu_dts_info ethosu_dts_info_##n = { \ .base_addr = (void *)DT_INST_REG_ADDR(n), \ .secure_enable = DT_INST_PROP(n, secure_enable), \ .privilege_enable = DT_INST_PROP(n, privilege_enable), \ .irq_config = ðosu_zephyr_irq_config_##n, \ }; \ \ DEVICE_DT_INST_DEFINE(n, ethosu_zephyr_init, NULL, ðosu_data_##n, ðosu_dts_info_##n, \ POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT, NULL); DT_INST_FOREACH_STATUS_OKAY(ETHOSU_DEVICE_INIT);