123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 |
- /*
- * Copyright (c) 2017-2024, Arm Limited and Contributors. All rights reserved.
- *
- * SPDX-License-Identifier: BSD-3-Clause
- */
- #include <assert.h>
- #include <arch_helpers.h>
- #include <common/debug.h>
- #include <drivers/arm/css/scmi.h>
- #include <drivers/delay_timer.h>
- #include "scmi_private.h"
- #if HW_ASSISTED_COHERENCY
- #define scmi_lock_init(lock)
- #define scmi_lock_get(lock) spin_lock(lock)
- #define scmi_lock_release(lock) spin_unlock(lock)
- #else
- #define scmi_lock_init(lock) bakery_lock_init(lock)
- #define scmi_lock_get(lock) bakery_lock_get(lock)
- #define scmi_lock_release(lock) bakery_lock_release(lock)
- #endif
- /*
- * Private helper function to get exclusive access to SCMI channel.
- */
- void scmi_get_channel(scmi_channel_t *ch)
- {
- assert(ch->lock);
- scmi_lock_get(ch->lock);
- /* Make sure any previous command has finished */
- assert(SCMI_IS_CHANNEL_FREE(
- ((mailbox_mem_t *)(ch->info->scmi_mbx_mem))->status));
- }
- /*
- * Private helper function to transfer ownership of channel from AP to SCP.
- */
- void scmi_send_sync_command(scmi_channel_t *ch)
- {
- mailbox_mem_t *mbx_mem = (mailbox_mem_t *)(ch->info->scmi_mbx_mem);
- SCMI_MARK_CHANNEL_BUSY(mbx_mem->status);
- /*
- * Ensure that any write to the SCMI payload area is seen by SCP before
- * we write to the doorbell register. If these 2 writes were reordered
- * by the CPU then SCP would read stale payload data
- */
- dmbst();
- ch->info->ring_doorbell(ch->info);
- /*
- * Ensure that the write to the doorbell register is ordered prior to
- * checking whether the channel is free.
- */
- dmbsy();
- /* Wait for channel to be free */
- while (!SCMI_IS_CHANNEL_FREE(mbx_mem->status)) {
- if (ch->info->delay != 0)
- udelay(ch->info->delay);
- }
- /*
- * Ensure that any read to the SCMI payload area is done after reading
- * mailbox status. If these 2 reads were reordered then the CPU would
- * read invalid payload data
- */
- dmbld();
- }
- /*
- * Private helper function to release exclusive access to SCMI channel.
- */
- void scmi_put_channel(scmi_channel_t *ch)
- {
- /* Make sure any previous command has finished */
- assert(SCMI_IS_CHANNEL_FREE(
- ((mailbox_mem_t *)(ch->info->scmi_mbx_mem))->status));
- assert(ch->lock);
- scmi_lock_release(ch->lock);
- }
- /*
- * API to query the SCMI protocol version.
- */
- int scmi_proto_version(void *p, uint32_t proto_id, uint32_t *version)
- {
- mailbox_mem_t *mbx_mem;
- unsigned int token = 0;
- int ret;
- scmi_channel_t *ch = (scmi_channel_t *)p;
- validate_scmi_channel(ch);
- scmi_get_channel(ch);
- mbx_mem = (mailbox_mem_t *)(ch->info->scmi_mbx_mem);
- mbx_mem->msg_header = SCMI_MSG_CREATE(proto_id, SCMI_PROTO_VERSION_MSG,
- token);
- mbx_mem->len = SCMI_PROTO_VERSION_MSG_LEN;
- mbx_mem->flags = SCMI_FLAG_RESP_POLL;
- scmi_send_sync_command(ch);
- /* Get the return values */
- SCMI_PAYLOAD_RET_VAL2(mbx_mem->payload, ret, *version);
- assert(mbx_mem->len == SCMI_PROTO_VERSION_RESP_LEN);
- assert(token == SCMI_MSG_GET_TOKEN(mbx_mem->msg_header));
- scmi_put_channel(ch);
- return ret;
- }
- /*
- * API to query the protocol message attributes for a SCMI protocol.
- */
- int scmi_proto_msg_attr(void *p, uint32_t proto_id,
- uint32_t command_id, uint32_t *attr)
- {
- mailbox_mem_t *mbx_mem;
- unsigned int token = 0;
- int ret;
- scmi_channel_t *ch = (scmi_channel_t *)p;
- validate_scmi_channel(ch);
- scmi_get_channel(ch);
- mbx_mem = (mailbox_mem_t *)(ch->info->scmi_mbx_mem);
- mbx_mem->msg_header = SCMI_MSG_CREATE(proto_id,
- SCMI_PROTO_MSG_ATTR_MSG, token);
- mbx_mem->len = SCMI_PROTO_MSG_ATTR_MSG_LEN;
- mbx_mem->flags = SCMI_FLAG_RESP_POLL;
- SCMI_PAYLOAD_ARG1(mbx_mem->payload, command_id);
- scmi_send_sync_command(ch);
- /* Get the return values */
- SCMI_PAYLOAD_RET_VAL2(mbx_mem->payload, ret, *attr);
- assert(mbx_mem->len == SCMI_PROTO_MSG_ATTR_RESP_LEN);
- assert(token == SCMI_MSG_GET_TOKEN(mbx_mem->msg_header));
- scmi_put_channel(ch);
- return ret;
- }
- /*
- * SCMI Driver initialization API. Returns initialized channel on success
- * or NULL on error. The return type is an opaque void pointer.
- */
- void *scmi_init(scmi_channel_t *ch)
- {
- uint32_t version;
- int ret;
- assert(ch && ch->info);
- assert(ch->info->db_reg_addr);
- assert(ch->info->db_modify_mask);
- assert(ch->info->db_preserve_mask);
- assert(ch->info->ring_doorbell != NULL);
- assert(ch->lock);
- scmi_lock_init(ch->lock);
- ch->is_initialized = 1;
- ret = scmi_proto_version(ch, SCMI_PWR_DMN_PROTO_ID, &version);
- if (ret != SCMI_E_SUCCESS) {
- WARN("SCMI power domain protocol version message failed\n");
- goto error;
- }
- if (!is_scmi_version_compatible(SCMI_PWR_DMN_PROTO_VER, version)) {
- WARN("SCMI power domain protocol version 0x%x incompatible with driver version 0x%x\n",
- version, SCMI_PWR_DMN_PROTO_VER);
- goto error;
- }
- VERBOSE("SCMI power domain protocol version 0x%x detected\n", version);
- ret = scmi_proto_version(ch, SCMI_SYS_PWR_PROTO_ID, &version);
- if ((ret != SCMI_E_SUCCESS)) {
- WARN("SCMI system power protocol version message failed\n");
- goto error;
- }
- if (!is_scmi_version_compatible(SCMI_SYS_PWR_PROTO_VER, version)) {
- WARN("SCMI system power management protocol version 0x%x incompatible with driver version 0x%x\n",
- version, SCMI_SYS_PWR_PROTO_VER);
- goto error;
- }
- VERBOSE("SCMI system power management protocol version 0x%x detected\n",
- version);
- INFO("SCMI driver initialized\n");
- return (void *)ch;
- error:
- ch->is_initialized = 0;
- return NULL;
- }
|