1ae8c8068SEtienne Carriere // SPDX-License-Identifier: BSD-3-Clause 2ae8c8068SEtienne Carriere /* 3ae8c8068SEtienne Carriere * Copyright (c) 2015-2019, Arm Limited and Contributors. All rights reserved. 45c34a982SEtienne Carriere * Copyright (c) 2019-2022, Linaro Limited 5ae8c8068SEtienne Carriere */ 65c34a982SEtienne Carriere #include <assert.h> 7ae8c8068SEtienne Carriere #include <drivers/scmi-msg.h> 8ae8c8068SEtienne Carriere #include <drivers/scmi.h> 94a382700SEtienne Carriere #include <kernel/misc.h> 105c34a982SEtienne Carriere #include <kernel/spinlock.h> 114a382700SEtienne Carriere #include <kernel/thread.h> 125c34a982SEtienne Carriere #include <string.h> 13ae8c8068SEtienne Carriere #include <trace.h> 14ae8c8068SEtienne Carriere 1560c96f68SEtienne Carriere #include "base.h" 1660c96f68SEtienne Carriere #include "clock.h" 17ae8c8068SEtienne Carriere #include "common.h" 1860c96f68SEtienne Carriere #include "reset_domain.h" 1960c96f68SEtienne Carriere #include "voltage_domain.h" 20ae8c8068SEtienne Carriere 214a382700SEtienne Carriere /* Provision input message payload buffers for each supported entry channel */ 224a382700SEtienne Carriere #define SCMI_PAYLOAD_U32_MAX (SCMI_SEC_PAYLOAD_SIZE / sizeof(uint32_t)) 234a382700SEtienne Carriere 244a382700SEtienne Carriere static uint32_t threaded_payload[CFG_NUM_THREADS][SCMI_PAYLOAD_U32_MAX] 254a382700SEtienne Carriere __maybe_unused; 264a382700SEtienne Carriere 274a382700SEtienne Carriere static uint32_t interrupt_payload[CFG_TEE_CORE_NB_CORE][SCMI_PAYLOAD_U32_MAX] 284a382700SEtienne Carriere __maybe_unused; 294a382700SEtienne Carriere 304a382700SEtienne Carriere static uint32_t fastcall_payload[CFG_TEE_CORE_NB_CORE][SCMI_PAYLOAD_U32_MAX] 314a382700SEtienne Carriere __maybe_unused; 324a382700SEtienne Carriere 335c34a982SEtienne Carriere /* SMP protection on channel->busy field */ 345c34a982SEtienne Carriere static unsigned int smt_channels_lock; 355c34a982SEtienne Carriere 365c34a982SEtienne Carriere /* If channel is not busy, set busy and return true, otherwise return false */ 375c34a982SEtienne Carriere bool scmi_msg_claim_channel(struct scmi_msg_channel *channel) 385c34a982SEtienne Carriere { 395c34a982SEtienne Carriere uint32_t exceptions = cpu_spin_lock_xsave(&smt_channels_lock); 405c34a982SEtienne Carriere bool channel_is_busy = channel->busy; 415c34a982SEtienne Carriere 425c34a982SEtienne Carriere if (!channel_is_busy) 435c34a982SEtienne Carriere channel->busy = true; 445c34a982SEtienne Carriere 455c34a982SEtienne Carriere cpu_spin_unlock_xrestore(&smt_channels_lock, exceptions); 465c34a982SEtienne Carriere 475c34a982SEtienne Carriere return !channel_is_busy; 485c34a982SEtienne Carriere } 495c34a982SEtienne Carriere 505c34a982SEtienne Carriere void scmi_msg_release_channel(struct scmi_msg_channel *channel) 515c34a982SEtienne Carriere { 525c34a982SEtienne Carriere channel->busy = false; 535c34a982SEtienne Carriere } 545c34a982SEtienne Carriere 55ae8c8068SEtienne Carriere void scmi_status_response(struct scmi_msg *msg, int32_t status) 56ae8c8068SEtienne Carriere { 57ae8c8068SEtienne Carriere assert(msg->out && msg->out_size >= sizeof(int32_t)); 58ae8c8068SEtienne Carriere 59ae8c8068SEtienne Carriere memcpy(msg->out, &status, sizeof(int32_t)); 60ae8c8068SEtienne Carriere msg->out_size_out = sizeof(int32_t); 61ae8c8068SEtienne Carriere } 62ae8c8068SEtienne Carriere 63ae8c8068SEtienne Carriere void scmi_write_response(struct scmi_msg *msg, void *payload, size_t size) 64ae8c8068SEtienne Carriere { 657b49ff33SEtienne Carriere if (msg->out_size < size) { 667b49ff33SEtienne Carriere DMSG("SCMI resp. payload %zu > %zu bytes", size, msg->out_size); 677b49ff33SEtienne Carriere scmi_status_response(msg, SCMI_PROTOCOL_ERROR); 687b49ff33SEtienne Carriere } else { 69ae8c8068SEtienne Carriere memcpy(msg->out, payload, size); 70ae8c8068SEtienne Carriere msg->out_size_out = size; 71ae8c8068SEtienne Carriere } 727b49ff33SEtienne Carriere } 73ae8c8068SEtienne Carriere 74ae8c8068SEtienne Carriere void scmi_process_message(struct scmi_msg *msg) 75ae8c8068SEtienne Carriere { 76ae8c8068SEtienne Carriere scmi_msg_handler_t handler = NULL; 77ae8c8068SEtienne Carriere 78ae8c8068SEtienne Carriere switch (msg->protocol_id) { 79ae8c8068SEtienne Carriere case SCMI_PROTOCOL_ID_BASE: 80ae8c8068SEtienne Carriere handler = scmi_msg_get_base_handler(msg); 81ae8c8068SEtienne Carriere break; 82a7a9e3baSEtienne Carriere case SCMI_PROTOCOL_ID_CLOCK: 83a7a9e3baSEtienne Carriere handler = scmi_msg_get_clock_handler(msg); 84a7a9e3baSEtienne Carriere break; 8556a1f10eSEtienne Carriere case SCMI_PROTOCOL_ID_RESET_DOMAIN: 8656a1f10eSEtienne Carriere handler = scmi_msg_get_rd_handler(msg); 8756a1f10eSEtienne Carriere break; 88006d89b8SEtienne Carriere case SCMI_PROTOCOL_ID_VOLTAGE_DOMAIN: 89006d89b8SEtienne Carriere handler = scmi_msg_get_voltd_handler(msg); 90006d89b8SEtienne Carriere break; 91ae8c8068SEtienne Carriere default: 92ae8c8068SEtienne Carriere break; 93ae8c8068SEtienne Carriere } 94ae8c8068SEtienne Carriere 95ae8c8068SEtienne Carriere if (handler) { 96ae8c8068SEtienne Carriere handler(msg); 97ae8c8068SEtienne Carriere return; 98ae8c8068SEtienne Carriere } 99ae8c8068SEtienne Carriere 100659a1f88SEtienne Carriere DMSG("Channel %u Protocol %#x Message %#x: not supported", 101659a1f88SEtienne Carriere msg->channel_id, msg->protocol_id, msg->message_id); 102ae8c8068SEtienne Carriere 103ae8c8068SEtienne Carriere scmi_status_response(msg, SCMI_NOT_SUPPORTED); 104ae8c8068SEtienne Carriere } 1054a382700SEtienne Carriere 1064a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_FASTCALL_ENTRY 1074a382700SEtienne Carriere void scmi_smt_fastcall_smc_entry(unsigned int channel_id) 1084a382700SEtienne Carriere { 1094a382700SEtienne Carriere assert(!plat_scmi_get_channel(channel_id)->threaded); 1104a382700SEtienne Carriere 1114a382700SEtienne Carriere scmi_entry_smt(channel_id, fastcall_payload[get_core_pos()]); 1124a382700SEtienne Carriere } 1134a382700SEtienne Carriere #endif 1144a382700SEtienne Carriere 1154a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_INTERRUPT_ENTRY 1164a382700SEtienne Carriere void scmi_smt_interrupt_entry(unsigned int channel_id) 1174a382700SEtienne Carriere { 1184a382700SEtienne Carriere assert(!plat_scmi_get_channel(channel_id)->threaded); 1194a382700SEtienne Carriere 1204a382700SEtienne Carriere scmi_entry_smt(channel_id, interrupt_payload[get_core_pos()]); 1214a382700SEtienne Carriere } 1224a382700SEtienne Carriere #endif 1234a382700SEtienne Carriere 1244a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_THREAD_ENTRY 1254a382700SEtienne Carriere void scmi_smt_threaded_entry(unsigned int channel_id) 1264a382700SEtienne Carriere { 1274a382700SEtienne Carriere assert(plat_scmi_get_channel(channel_id)->threaded); 1284a382700SEtienne Carriere 1294a382700SEtienne Carriere scmi_entry_smt(channel_id, threaded_payload[thread_get_id()]); 1304a382700SEtienne Carriere } 1314a382700SEtienne Carriere #endif 132*916cc52aSEtienne Carriere 133*916cc52aSEtienne Carriere #ifdef CFG_SCMI_MSG_SHM_MSG 134*916cc52aSEtienne Carriere TEE_Result scmi_msg_threaded_entry(unsigned int channel_id, 135*916cc52aSEtienne Carriere void *in_buf, size_t in_size, 136*916cc52aSEtienne Carriere void *out_buf, size_t *out_size) 137*916cc52aSEtienne Carriere { 138*916cc52aSEtienne Carriere assert(plat_scmi_get_channel(channel_id)->threaded); 139*916cc52aSEtienne Carriere 140*916cc52aSEtienne Carriere return scmi_entry_msg(channel_id, in_buf, in_size, out_buf, out_size, 141*916cc52aSEtienne Carriere threaded_payload[thread_get_id()]); 142*916cc52aSEtienne Carriere } 143*916cc52aSEtienne Carriere #endif 144