Added more APIs for ARM SCMI clock management protocol. - scmi_clock_rate_set - scmi_clock_parent_get - scmi_clock_parent_set Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
275 lines
5.5 KiB
C
275 lines
5.5 KiB
C
/*
|
|
* Copyright 2024 NXP
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <zephyr/drivers/firmware/scmi/clk.h>
|
|
#include <string.h>
|
|
|
|
/* TODO: if extended attributes are supported this should be moved
|
|
* to the header file so that users will have access to it.
|
|
*/
|
|
#define SCMI_CLK_CONFIG_EA_MASK GENMASK(23, 16)
|
|
|
|
struct scmi_clock_attributes_reply {
|
|
int32_t status;
|
|
uint32_t attributes;
|
|
};
|
|
|
|
struct scmi_clock_rate_set_reply {
|
|
int32_t status;
|
|
uint32_t rate[2];
|
|
};
|
|
|
|
struct scmi_clock_parent_get_reply {
|
|
int32_t status;
|
|
uint32_t parent_id;
|
|
};
|
|
|
|
struct scmi_clock_parent_config {
|
|
uint32_t clk_id;
|
|
uint32_t parent_id;
|
|
};
|
|
|
|
int scmi_clock_rate_get(struct scmi_protocol *proto,
|
|
uint32_t clk_id, uint32_t *rate)
|
|
{
|
|
struct scmi_message msg, reply;
|
|
int ret;
|
|
struct scmi_clock_rate_set_reply reply_buffer;
|
|
|
|
/* sanity checks */
|
|
if (!proto || !rate) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
msg.hdr = SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_CLOCK_RATE_GET,
|
|
SCMI_COMMAND, proto->id, 0x0);
|
|
msg.len = sizeof(clk_id);
|
|
msg.content = &clk_id;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(reply_buffer);
|
|
reply.content = &reply_buffer;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (reply_buffer.status != SCMI_SUCCESS) {
|
|
return scmi_status_to_errno(reply_buffer.status);
|
|
}
|
|
|
|
*rate = reply_buffer.rate[0];
|
|
|
|
return 0;
|
|
}
|
|
|
|
int scmi_clock_rate_set(struct scmi_protocol *proto, struct scmi_clock_rate_config *cfg)
|
|
{
|
|
struct scmi_message msg, reply;
|
|
int status, ret;
|
|
|
|
/* sanity checks */
|
|
if (!proto || !cfg) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Currently ASYNC flag is not supported. */
|
|
if (cfg->flags & SCMI_CLK_RATE_SET_FLAGS_ASYNC) {
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
msg.hdr = SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_CLOCK_RATE_SET, SCMI_COMMAND, proto->id, 0x0);
|
|
msg.len = sizeof(*cfg);
|
|
msg.content = cfg;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(status);
|
|
reply.content = &status;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (status != SCMI_SUCCESS) {
|
|
return scmi_status_to_errno(status);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int scmi_clock_parent_get(struct scmi_protocol *proto, uint32_t clk_id, uint32_t *parent_id)
|
|
{
|
|
struct scmi_message msg, reply;
|
|
int ret;
|
|
struct scmi_clock_parent_get_reply reply_buffer;
|
|
|
|
/* sanity checks */
|
|
if (!proto || !parent_id) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
msg.hdr =
|
|
SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_CLOCK_PARENT_GET, SCMI_COMMAND, proto->id, 0x0);
|
|
msg.len = sizeof(clk_id);
|
|
msg.content = &clk_id;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(reply_buffer);
|
|
reply.content = &reply_buffer;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (reply_buffer.status != SCMI_SUCCESS) {
|
|
return scmi_status_to_errno(reply_buffer.status);
|
|
}
|
|
|
|
*parent_id = reply_buffer.parent_id;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int scmi_clock_parent_set(struct scmi_protocol *proto, uint32_t clk_id, uint32_t parent_id)
|
|
{
|
|
struct scmi_clock_parent_config cfg = {.clk_id = clk_id, .parent_id = parent_id};
|
|
struct scmi_message msg, reply;
|
|
int status, ret;
|
|
|
|
/* sanity checks */
|
|
if (!proto) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
msg.hdr =
|
|
SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_CLOCK_PARENT_SET, SCMI_COMMAND, proto->id, 0x0);
|
|
msg.len = sizeof(cfg);
|
|
msg.content = &cfg;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(status);
|
|
reply.content = &status;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (status != SCMI_SUCCESS) {
|
|
return scmi_status_to_errno(status);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int scmi_clock_config_set(struct scmi_protocol *proto,
|
|
struct scmi_clock_config *cfg)
|
|
{
|
|
struct scmi_message msg, reply;
|
|
int status, ret;
|
|
|
|
/* sanity checks */
|
|
if (!proto || !cfg) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* extended attributes currently not supported */
|
|
if (cfg->attributes & SCMI_CLK_CONFIG_EA_MASK) {
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
/* invalid because extended attributes are not supported */
|
|
if (SCMI_CLK_CONFIG_ENABLE_DISABLE(cfg->attributes) == 3) {
|
|
return -ENOTSUP;
|
|
}
|
|
|
|
/* this is a reserved value */
|
|
if (SCMI_CLK_CONFIG_ENABLE_DISABLE(cfg->attributes) == 2) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
msg.hdr = SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_CLOCK_CONFIG_SET,
|
|
SCMI_COMMAND, proto->id, 0x0);
|
|
msg.len = sizeof(*cfg);
|
|
msg.content = cfg;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(status);
|
|
reply.content = &status;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (status != SCMI_SUCCESS) {
|
|
return scmi_status_to_errno(status);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int scmi_clock_protocol_attributes(struct scmi_protocol *proto, uint32_t *attributes)
|
|
{
|
|
struct scmi_message msg, reply;
|
|
struct scmi_clock_attributes_reply reply_buffer;
|
|
int ret;
|
|
|
|
/* sanity checks */
|
|
if (!proto || !attributes) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (proto->id != SCMI_PROTOCOL_CLOCK) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
msg.hdr = SCMI_MESSAGE_HDR_MAKE(SCMI_CLK_MSG_PROTOCOL_ATTRIBUTES,
|
|
SCMI_COMMAND, proto->id, 0x0);
|
|
/* command has no parameters */
|
|
msg.len = 0x0;
|
|
msg.content = NULL;
|
|
|
|
reply.hdr = msg.hdr;
|
|
reply.len = sizeof(reply_buffer);
|
|
reply.content = &reply_buffer;
|
|
|
|
ret = scmi_send_message(proto, &msg, &reply);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (reply_buffer.status != 0) {
|
|
return scmi_status_to_errno(reply_buffer.status);
|
|
}
|
|
|
|
*attributes = reply_buffer.attributes;
|
|
|
|
return 0;
|
|
}
|