Commit 5c4ba3cc authored by Sudeep Holla's avatar Sudeep Holla

firmware: arm_scmi: add option for polling based performance domain operations

In order to implement fast CPU DVFS switching, we need to perform all
DVFS operations atomically. Since SCMI transfer already provide option
to choose between pooling vs interrupt driven(default), we can opt for
polling based transfers for set,get performance domain operations.

This patch adds option to choose between polling vs interrupt driven
SCMI transfers for set,get performance level operations.

Cc: Arnd Bergmann <arnd@arndb.de>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarSudeep Holla <sudeep.holla@arm.com>
parent d4c3751a
...@@ -292,8 +292,8 @@ static int scmi_perf_limits_get(const struct scmi_handle *handle, u32 domain, ...@@ -292,8 +292,8 @@ static int scmi_perf_limits_get(const struct scmi_handle *handle, u32 domain,
return ret; return ret;
} }
static int static int scmi_perf_level_set(const struct scmi_handle *handle, u32 domain,
scmi_perf_level_set(const struct scmi_handle *handle, u32 domain, u32 level) u32 level, bool poll)
{ {
int ret; int ret;
struct scmi_xfer *t; struct scmi_xfer *t;
...@@ -304,6 +304,7 @@ scmi_perf_level_set(const struct scmi_handle *handle, u32 domain, u32 level) ...@@ -304,6 +304,7 @@ scmi_perf_level_set(const struct scmi_handle *handle, u32 domain, u32 level)
if (ret) if (ret)
return ret; return ret;
t->hdr.poll_completion = poll;
lvl = t->tx.buf; lvl = t->tx.buf;
lvl->domain = cpu_to_le32(domain); lvl->domain = cpu_to_le32(domain);
lvl->level = cpu_to_le32(level); lvl->level = cpu_to_le32(level);
...@@ -314,8 +315,8 @@ scmi_perf_level_set(const struct scmi_handle *handle, u32 domain, u32 level) ...@@ -314,8 +315,8 @@ scmi_perf_level_set(const struct scmi_handle *handle, u32 domain, u32 level)
return ret; return ret;
} }
static int static int scmi_perf_level_get(const struct scmi_handle *handle, u32 domain,
scmi_perf_level_get(const struct scmi_handle *handle, u32 domain, u32 *level) u32 *level, bool poll)
{ {
int ret; int ret;
struct scmi_xfer *t; struct scmi_xfer *t;
...@@ -325,6 +326,7 @@ scmi_perf_level_get(const struct scmi_handle *handle, u32 domain, u32 *level) ...@@ -325,6 +326,7 @@ scmi_perf_level_get(const struct scmi_handle *handle, u32 domain, u32 *level)
if (ret) if (ret)
return ret; return ret;
t->hdr.poll_completion = poll;
*(__le32 *)t->tx.buf = cpu_to_le32(domain); *(__le32 *)t->tx.buf = cpu_to_le32(domain);
ret = scmi_do_xfer(handle, t); ret = scmi_do_xfer(handle, t);
...@@ -400,23 +402,24 @@ static int scmi_dvfs_get_transition_latency(const struct scmi_handle *handle, ...@@ -400,23 +402,24 @@ static int scmi_dvfs_get_transition_latency(const struct scmi_handle *handle,
} }
static int scmi_dvfs_freq_set(const struct scmi_handle *handle, u32 domain, static int scmi_dvfs_freq_set(const struct scmi_handle *handle, u32 domain,
unsigned long freq) unsigned long freq, bool poll)
{ {
struct scmi_perf_info *pi = handle->perf_priv; struct scmi_perf_info *pi = handle->perf_priv;
struct perf_dom_info *dom = pi->dom_info + domain; struct perf_dom_info *dom = pi->dom_info + domain;
return scmi_perf_level_set(handle, domain, freq / dom->mult_factor); return scmi_perf_level_set(handle, domain, freq / dom->mult_factor,
poll);
} }
static int scmi_dvfs_freq_get(const struct scmi_handle *handle, u32 domain, static int scmi_dvfs_freq_get(const struct scmi_handle *handle, u32 domain,
unsigned long *freq) unsigned long *freq, bool poll)
{ {
int ret; int ret;
u32 level; u32 level;
struct scmi_perf_info *pi = handle->perf_priv; struct scmi_perf_info *pi = handle->perf_priv;
struct perf_dom_info *dom = pi->dom_info + domain; struct perf_dom_info *dom = pi->dom_info + domain;
ret = scmi_perf_level_get(handle, domain, &level); ret = scmi_perf_level_get(handle, domain, &level, poll);
if (!ret) if (!ret)
*freq = level * dom->mult_factor; *freq = level * dom->mult_factor;
......
...@@ -98,18 +98,18 @@ struct scmi_perf_ops { ...@@ -98,18 +98,18 @@ struct scmi_perf_ops {
int (*limits_get)(const struct scmi_handle *handle, u32 domain, int (*limits_get)(const struct scmi_handle *handle, u32 domain,
u32 *max_perf, u32 *min_perf); u32 *max_perf, u32 *min_perf);
int (*level_set)(const struct scmi_handle *handle, u32 domain, int (*level_set)(const struct scmi_handle *handle, u32 domain,
u32 level); u32 level, bool poll);
int (*level_get)(const struct scmi_handle *handle, u32 domain, int (*level_get)(const struct scmi_handle *handle, u32 domain,
u32 *level); u32 *level, bool poll);
int (*device_domain_id)(struct device *dev); int (*device_domain_id)(struct device *dev);
int (*get_transition_latency)(const struct scmi_handle *handle, int (*get_transition_latency)(const struct scmi_handle *handle,
struct device *dev); struct device *dev);
int (*add_opps_to_device)(const struct scmi_handle *handle, int (*add_opps_to_device)(const struct scmi_handle *handle,
struct device *dev); struct device *dev);
int (*freq_set)(const struct scmi_handle *handle, u32 domain, int (*freq_set)(const struct scmi_handle *handle, u32 domain,
unsigned long rate); unsigned long rate, bool poll);
int (*freq_get)(const struct scmi_handle *handle, u32 domain, int (*freq_get)(const struct scmi_handle *handle, u32 domain,
unsigned long *rate); unsigned long *rate, bool poll);
}; };
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment