xref: /optee_os/core/drivers/scmi-msg/entry.c (revision 4a3827000583c094d5fc365150d31ffb02259a67)
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>
9*4a382700SEtienne Carriere #include <kernel/misc.h>
105c34a982SEtienne Carriere #include <kernel/spinlock.h>
11*4a382700SEtienne 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 
21*4a382700SEtienne Carriere /* Provision input message payload buffers for each supported entry channel */
22*4a382700SEtienne Carriere #define SCMI_PAYLOAD_U32_MAX	(SCMI_SEC_PAYLOAD_SIZE / sizeof(uint32_t))
23*4a382700SEtienne Carriere 
24*4a382700SEtienne Carriere static uint32_t threaded_payload[CFG_NUM_THREADS][SCMI_PAYLOAD_U32_MAX]
25*4a382700SEtienne Carriere __maybe_unused;
26*4a382700SEtienne Carriere 
27*4a382700SEtienne Carriere static uint32_t interrupt_payload[CFG_TEE_CORE_NB_CORE][SCMI_PAYLOAD_U32_MAX]
28*4a382700SEtienne Carriere __maybe_unused;
29*4a382700SEtienne Carriere 
30*4a382700SEtienne Carriere static uint32_t fastcall_payload[CFG_TEE_CORE_NB_CORE][SCMI_PAYLOAD_U32_MAX]
31*4a382700SEtienne Carriere __maybe_unused;
32*4a382700SEtienne 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 }
105*4a382700SEtienne Carriere 
106*4a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_FASTCALL_ENTRY
107*4a382700SEtienne Carriere void scmi_smt_fastcall_smc_entry(unsigned int channel_id)
108*4a382700SEtienne Carriere {
109*4a382700SEtienne Carriere 	assert(!plat_scmi_get_channel(channel_id)->threaded);
110*4a382700SEtienne Carriere 
111*4a382700SEtienne Carriere 	scmi_entry_smt(channel_id, fastcall_payload[get_core_pos()]);
112*4a382700SEtienne Carriere }
113*4a382700SEtienne Carriere #endif
114*4a382700SEtienne Carriere 
115*4a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_INTERRUPT_ENTRY
116*4a382700SEtienne Carriere void scmi_smt_interrupt_entry(unsigned int channel_id)
117*4a382700SEtienne Carriere {
118*4a382700SEtienne Carriere 	assert(!plat_scmi_get_channel(channel_id)->threaded);
119*4a382700SEtienne Carriere 
120*4a382700SEtienne Carriere 	scmi_entry_smt(channel_id, interrupt_payload[get_core_pos()]);
121*4a382700SEtienne Carriere }
122*4a382700SEtienne Carriere #endif
123*4a382700SEtienne Carriere 
124*4a382700SEtienne Carriere #ifdef CFG_SCMI_MSG_SMT_THREAD_ENTRY
125*4a382700SEtienne Carriere void scmi_smt_threaded_entry(unsigned int channel_id)
126*4a382700SEtienne Carriere {
127*4a382700SEtienne Carriere 	assert(plat_scmi_get_channel(channel_id)->threaded);
128*4a382700SEtienne Carriere 
129*4a382700SEtienne Carriere 	scmi_entry_smt(channel_id, threaded_payload[thread_get_id()]);
130*4a382700SEtienne Carriere }
131*4a382700SEtienne Carriere #endif
132