// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2015-2019, Arm Limited and Contributors. All rights reserved. * Copyright (C) 2019-2022 Linaro Limited. */ #define LOG_CATEGORY UCLASS_SCMI_AGENT #include #include #include #include #include #include #include #include #include #include #include #include "smt.h" static void scmi_smt_enable_intr(struct scmi_smt *smt, bool enable) { struct scmi_smt_header *hdr = (void *)smt->buf; if (enable) iowrite32(ioread32(&hdr->flags) | SCMI_SHMEM_FLAG_INTR_ENABLED, &hdr->flags); else iowrite32(ioread32(&hdr->flags) & ~SCMI_SHMEM_FLAG_INTR_ENABLED, &hdr->flags); } /** * Get shared memory configuration defined by the referred DT phandle * Return with a errno compliant value. */ int scmi_dt_get_smt_buffer(struct udevice *dev, struct scmi_smt *smt) { int ret; struct ofnode_phandle_args args; struct resource resource; ret = dev_read_phandle_with_args(dev, "shmem", NULL, 0, 0, &args); if (ret) return ret; ret = ofnode_read_resource(args.node, 0, &resource); if (ret) return ret; smt->size = resource_size(&resource); if (smt->size < sizeof(struct scmi_smt_header)) { dev_err(dev, "Shared memory buffer too small\n"); return -EINVAL; } smt->buf = devm_ioremap(dev, resource.start, smt->size); if (!smt->buf) return -ENOMEM; if (device_is_compatible(dev, "arm,scmi") && ofnode_has_property(dev_ofnode(dev), "mboxes")) scmi_smt_enable_intr(smt, true); #ifdef CONFIG_ARM if (dcache_status()) { u32 align_size; if (IS_ENABLED(CONFIG_ARM64)) align_size = PAGE_SIZE; else align_size = MMU_SECTION_SIZE; mmu_set_region_dcache_behaviour(ALIGN_DOWN((uintptr_t)smt->buf, align_size), ALIGN(smt->size, align_size), DCACHE_OFF); } #endif return 0; } /** * Write SCMI message @msg into a SMT shared buffer @smt. * Return 0 on success and with a negative errno in case of error. */ int scmi_write_msg_to_smt(struct udevice *dev, struct scmi_smt *smt, struct scmi_msg *msg) { struct scmi_smt_header *hdr = (void *)smt->buf; if ((!msg->in_msg && msg->in_msg_sz) || (!msg->out_msg && msg->out_msg_sz)) return -EINVAL; if (!(ioread32(&hdr->channel_status) & SCMI_SHMEM_CHAN_STAT_CHANNEL_FREE)) { dev_dbg(dev, "Channel busy\n"); return -EBUSY; } if (smt->size < (sizeof(*hdr) + msg->in_msg_sz) || smt->size < (sizeof(*hdr) + msg->out_msg_sz)) { dev_err(dev, "Buffer write too small: mst->size:%zu, in_msg_sz:%zu, out_msg_sz:%zu\n", smt->size, msg->in_msg_sz, msg->out_msg_sz); return -ETOOSMALL; } /* Load message in shared memory */ iowrite32(ioread32(&hdr->channel_status) & ~SCMI_SHMEM_CHAN_STAT_CHANNEL_FREE, &hdr->channel_status); iowrite32(msg->in_msg_sz + sizeof(hdr->msg_header), &hdr->length); iowrite32(SMT_HEADER_TOKEN(0) | SMT_HEADER_MESSAGE_TYPE(0) | SMT_HEADER_PROTOCOL_ID(msg->protocol_id) | SMT_HEADER_MESSAGE_ID(msg->message_id), &hdr->msg_header); memcpy_toio(hdr->msg_payload, msg->in_msg, msg->in_msg_sz); return 0; } /** * Read SCMI message from a SMT shared buffer @smt and copy it into @msg. * Return 0 on success and with a negative errno in case of error. */ int scmi_read_resp_from_smt(struct udevice *dev, struct scmi_smt *smt, struct scmi_msg *msg) { struct scmi_smt_header *hdr = (void *)smt->buf; if (!(ioread32(&hdr->channel_status) & SCMI_SHMEM_CHAN_STAT_CHANNEL_FREE)) { dev_err(dev, "Channel unexpectedly busy\n"); return -EBUSY; } if (ioread32(&hdr->channel_status) & SCMI_SHMEM_CHAN_STAT_CHANNEL_ERROR) { dev_err(dev, "Channel error reported, reset channel\n"); return -ECOMM; } if (ioread32(&hdr->length) > msg->out_msg_sz + sizeof(hdr->msg_header)) { dev_err(dev, "Buffer too small: hdr->length:%u, out_msg_sz:%zu\n", ioread32(&hdr->length), msg->out_msg_sz); return -ETOOSMALL; } /* Get the data */ msg->out_msg_sz = ioread32(&hdr->length) - sizeof(hdr->msg_header); memcpy_fromio(msg->out_msg, hdr->msg_payload, msg->out_msg_sz); return 0; } /** * Clear SMT flags in shared buffer to allow further message exchange */ void scmi_clear_smt_channel(struct scmi_smt *smt) { struct scmi_smt_header *hdr = (void *)smt->buf; iowrite32(ioread32(&hdr->channel_status) & ~SCMI_SHMEM_CHAN_STAT_CHANNEL_ERROR, &hdr->channel_status); } /** * Write SCMI message @msg into a SMT_MSG shared buffer @smt. * Return 0 on success and with a negative errno in case of error. */ int scmi_msg_to_smt_msg(struct udevice *dev, struct scmi_smt *smt, struct scmi_msg *msg, size_t *buf_size) { struct scmi_smt_msg_header *hdr = (void *)smt->buf; if ((!msg->in_msg && msg->in_msg_sz) || (!msg->out_msg && msg->out_msg_sz)) return -EINVAL; if (smt->size < (sizeof(*hdr) + msg->in_msg_sz) || smt->size < (sizeof(*hdr) + msg->out_msg_sz)) { dev_err(dev, "Buffer too small: mst->size:%zu, in_msg_sz:%zu, out_msg_sz:%zu\n", smt->size, msg->in_msg_sz, msg->out_msg_sz); return -ETOOSMALL; } *buf_size = msg->in_msg_sz + sizeof(hdr->msg_header); iowrite32(SMT_HEADER_TOKEN(0) | SMT_HEADER_MESSAGE_TYPE(0) | SMT_HEADER_PROTOCOL_ID(msg->protocol_id) | SMT_HEADER_MESSAGE_ID(msg->message_id), &hdr->msg_header); memcpy_fromio(hdr->msg_payload, msg->in_msg, msg->in_msg_sz); return 0; } /** * Read SCMI message from a SMT shared buffer @smt and copy it into @msg. * Return 0 on success and with a negative errno in case of error. */ int scmi_msg_from_smt_msg(struct udevice *dev, struct scmi_smt *smt, struct scmi_msg *msg, size_t buf_size) { struct scmi_smt_msg_header *hdr = (void *)smt->buf; if (buf_size > msg->out_msg_sz + sizeof(hdr->msg_header)) { dev_err(dev, "Buffer too small: buf_size:%zu, out_msg_sz:%zu\n", buf_size, msg->out_msg_sz); return -ETOOSMALL; } msg->out_msg_sz = buf_size - sizeof(hdr->msg_header); memcpy_toio(msg->out_msg, hdr->msg_payload, msg->out_msg_sz); return 0; }