| /* Copyright (c) 2016-2017, The Linux Foundation. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 and |
| * only version 2 as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| */ |
| |
| #include <linux/atomic.h> |
| #include <linux/delay.h> |
| #include <linux/interrupt.h> |
| #include <linux/kernel.h> |
| #include <linux/list.h> |
| #include <linux/mailbox_client.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/platform_device.h> |
| #include <linux/slab.h> |
| #include <linux/spinlock.h> |
| #include <linux/types.h> |
| #include <linux/wait.h> |
| |
| #include <soc/qcom/rpmh.h> |
| #include <soc/qcom/tcs.h> |
| #include <soc/qcom/cmd-db.h> |
| |
| #define RPMH_MAX_MBOXES 2 |
| #define RPMH_MAX_FAST_RES 32 |
| #define RPMH_MAX_REQ_IN_BATCH 10 |
| |
| #define DEFINE_RPMH_MSG_ONSTACK(rc, s, q, c, name) \ |
| struct rpmh_msg name = { \ |
| .msg = { 0 }, \ |
| .msg.state = s, \ |
| .msg.is_complete = true, \ |
| .msg.payload = name.cmd, \ |
| .msg.num_payload = 0, \ |
| .cmd = { { 0 } }, \ |
| .waitq = q, \ |
| .wait_count = c, \ |
| .rc = rc, \ |
| .bit = -1, \ |
| } |
| |
| struct rpmh_req { |
| u32 addr; |
| u32 sleep_val; |
| u32 wake_val; |
| struct list_head list; |
| }; |
| |
| struct rpmh_msg { |
| struct tcs_mbox_msg msg; |
| struct tcs_cmd cmd[MAX_RPMH_PAYLOAD]; |
| wait_queue_head_t *waitq; |
| atomic_t *wait_count; |
| struct rpmh_client *rc; |
| int bit; |
| int err; /* relay error from mbox for sync calls */ |
| }; |
| |
| struct rpmh_mbox { |
| struct device_node *mbox_dn; |
| struct list_head resources; |
| spinlock_t lock; |
| struct rpmh_msg *msg_pool; |
| DECLARE_BITMAP(fast_req, RPMH_MAX_FAST_RES); |
| bool dirty; |
| bool in_solver_mode; |
| }; |
| |
| struct rpmh_client { |
| struct device *dev; |
| struct mbox_client client; |
| struct mbox_chan *chan; |
| struct rpmh_mbox *rpmh; |
| }; |
| |
| static struct rpmh_mbox mbox_ctrlr[RPMH_MAX_MBOXES]; |
| DEFINE_MUTEX(rpmh_mbox_mutex); |
| bool rpmh_standalone; |
| |
| static struct rpmh_msg *get_msg_from_pool(struct rpmh_client *rc) |
| { |
| struct rpmh_mbox *rpm = rc->rpmh; |
| struct rpmh_msg *msg = NULL; |
| int pos; |
| unsigned long flags; |
| |
| spin_lock_irqsave(&rpm->lock, flags); |
| pos = find_first_zero_bit(rpm->fast_req, RPMH_MAX_FAST_RES); |
| if (pos != RPMH_MAX_FAST_RES) { |
| bitmap_set(rpm->fast_req, pos, 1); |
| msg = &rpm->msg_pool[pos]; |
| memset(msg, 0, sizeof(*msg)); |
| msg->bit = pos; |
| msg->rc = rc; |
| } |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| return msg; |
| } |
| |
| static void rpmh_rx_cb(struct mbox_client *cl, void *msg) |
| { |
| struct rpmh_msg *rpm_msg = container_of(msg, struct rpmh_msg, msg); |
| |
| atomic_dec(rpm_msg->wait_count); |
| wake_up(rpm_msg->waitq); |
| } |
| |
| static void rpmh_tx_done(struct mbox_client *cl, void *msg, int r) |
| { |
| struct rpmh_msg *rpm_msg = container_of(msg, struct rpmh_msg, msg); |
| struct rpmh_mbox *rpm = rpm_msg->rc->rpmh; |
| atomic_t *wc = rpm_msg->wait_count; |
| wait_queue_head_t *waitq = rpm_msg->waitq; |
| unsigned long flags; |
| |
| rpm_msg->err = r; |
| |
| if (r) { |
| dev_err(rpm_msg->rc->dev, |
| "RPMH TX fail in msg addr 0x%x, err=%d\n", |
| rpm_msg->msg.payload[0].addr, r); |
| /* |
| * If we fail TX for a read, call then we won't get |
| * a rx_callback. Force a rx_cb. |
| */ |
| if (rpm_msg->msg.is_read) |
| rpmh_rx_cb(cl, msg); |
| } |
| |
| /* |
| * Copy the child object pointers before freeing up the parent, |
| * This way even if the parent (rpm_msg) object gets reused, we |
| * can free up the child objects (wq/wc) parallely. |
| * If you free up the children before the parent, then we run |
| * into an issue that the stack allocated parent object may be |
| * invalid before we can check the ->bit value. |
| */ |
| |
| /* If we allocated the pool, set it as available */ |
| if (rpm_msg->bit >= 0 && rpm_msg->bit != RPMH_MAX_FAST_RES) { |
| spin_lock_irqsave(&rpm->lock, flags); |
| bitmap_clear(rpm->fast_req, rpm_msg->bit, 1); |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| } |
| |
| /* Signal the blocking thread we are done */ |
| if (wc && atomic_dec_and_test(wc)) |
| if (waitq) |
| wake_up(waitq); |
| } |
| |
| static struct rpmh_req *__find_req(struct rpmh_client *rc, u32 addr) |
| { |
| struct rpmh_req *p, *req = NULL; |
| |
| list_for_each_entry(p, &rc->rpmh->resources, list) { |
| if (p->addr == addr) { |
| req = p; |
| break; |
| } |
| } |
| |
| return req; |
| } |
| |
| static struct rpmh_req *cache_rpm_request(struct rpmh_client *rc, |
| enum rpmh_state state, struct tcs_cmd *cmd) |
| { |
| struct rpmh_req *req; |
| struct rpmh_mbox *rpm = rc->rpmh; |
| unsigned long flags; |
| |
| spin_lock_irqsave(&rpm->lock, flags); |
| req = __find_req(rc, cmd->addr); |
| if (req) |
| goto existing; |
| |
| req = kzalloc(sizeof(*req), GFP_ATOMIC); |
| if (!req) { |
| req = ERR_PTR(-ENOMEM); |
| goto unlock; |
| } |
| |
| req->addr = cmd->addr; |
| req->sleep_val = req->wake_val = UINT_MAX; |
| INIT_LIST_HEAD(&req->list); |
| list_add_tail(&req->list, &rpm->resources); |
| |
| existing: |
| switch (state) { |
| case RPMH_ACTIVE_ONLY_STATE: |
| case RPMH_AWAKE_STATE: |
| if (req->sleep_val != UINT_MAX) |
| req->wake_val = cmd->data; |
| break; |
| case RPMH_WAKE_ONLY_STATE: |
| req->wake_val = cmd->data; |
| break; |
| case RPMH_SLEEP_STATE: |
| req->sleep_val = cmd->data; |
| break; |
| default: |
| break; |
| }; |
| |
| unlock: |
| rpm->dirty = true; |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| return req; |
| } |
| |
| /** |
| * __rpmh_write: Cache and send the RPMH request |
| * |
| * @rc: The RPMH client |
| * @state: Active/Sleep request type |
| * @rpm_msg: The data that needs to be sent (payload). |
| * |
| * Cache the RPMH request and send if the state is ACTIVE_ONLY. |
| * SLEEP/WAKE_ONLY requests are not sent to the controller at |
| * this time. Use rpmh_flush() to send them to the controller. |
| */ |
| int __rpmh_write(struct rpmh_client *rc, enum rpmh_state state, |
| struct rpmh_msg *rpm_msg) |
| { |
| struct rpmh_req *req; |
| int ret = 0; |
| int i; |
| |
| /* Cache the request in our store and link the payload */ |
| for (i = 0; i < rpm_msg->msg.num_payload; i++) { |
| req = cache_rpm_request(rc, state, &rpm_msg->msg.payload[i]); |
| if (IS_ERR(req)) |
| return PTR_ERR(req); |
| } |
| |
| rpm_msg->msg.state = state; |
| |
| /* Send to mailbox only if active or awake */ |
| if (state == RPMH_ACTIVE_ONLY_STATE || state == RPMH_AWAKE_STATE) { |
| ret = mbox_send_message(rc->chan, &rpm_msg->msg); |
| if (ret > 0) |
| ret = 0; |
| } else { |
| /* Clean up our call by spoofing tx_done */ |
| rpmh_tx_done(&rc->client, &rpm_msg->msg, ret); |
| } |
| |
| return ret; |
| } |
| |
| /** |
| * rpmh_write_single_async: Write a single RPMH command |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @state: Active/sleep set |
| * @addr: The ePCB address |
| * @data: The data |
| * |
| * Write a single value in fast-path. Fire and forget. |
| * May be called from atomic contexts. |
| */ |
| int rpmh_write_single_async(struct rpmh_client *rc, enum rpmh_state state, |
| u32 addr, u32 data) |
| { |
| struct rpmh_msg *rpm_msg; |
| |
| if (IS_ERR_OR_NULL(rc)) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm_msg = get_msg_from_pool(rc); |
| if (!rpm_msg) |
| return -ENOMEM; |
| |
| rpm_msg->cmd[0].addr = addr; |
| rpm_msg->cmd[0].data = data; |
| |
| rpm_msg->msg.payload = rpm_msg->cmd; |
| rpm_msg->msg.num_payload = 1; |
| |
| return __rpmh_write(rc, state, rpm_msg); |
| } |
| EXPORT_SYMBOL(rpmh_write_single_async); |
| |
| /** |
| * rpmh_write_single: Write a single RPMH command and |
| * wait for completion of the command. |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @state: Active/sleep set |
| * @addr: The ePCB address |
| * @offset: Offset of the resource |
| * @data: The data |
| * |
| * Write a single value in slow-path and wait for the request to be |
| * complete. Blocks until the request is completed on the accelerator. |
| * Do not call from atomic contexts. |
| */ |
| int rpmh_write_single(struct rpmh_client *rc, enum rpmh_state state, |
| u32 addr, u32 data) |
| { |
| DECLARE_WAIT_QUEUE_HEAD_ONSTACK(waitq); |
| atomic_t wait_count = ATOMIC_INIT(1); |
| DEFINE_RPMH_MSG_ONSTACK(rc, state, &waitq, &wait_count, rpm_msg); |
| int ret; |
| |
| if (IS_ERR_OR_NULL(rc)) |
| return -EINVAL; |
| |
| might_sleep(); |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm_msg.cmd[0].addr = addr; |
| rpm_msg.cmd[0].data = data; |
| rpm_msg.msg.num_payload = 1; |
| |
| ret = __rpmh_write(rc, state, &rpm_msg); |
| if (ret < 0) |
| return ret; |
| |
| wait_event(waitq, atomic_read(&wait_count) == 0); |
| |
| return rpm_msg.err; |
| } |
| EXPORT_SYMBOL(rpmh_write_single); |
| |
| struct rpmh_msg *__get_rpmh_msg_async(struct rpmh_client *rc, |
| enum rpmh_state state, struct tcs_cmd *cmd, int n) |
| { |
| struct rpmh_msg *rpm_msg; |
| |
| if (IS_ERR_OR_NULL(rc) || !cmd || n <= 0 || n > MAX_RPMH_PAYLOAD) |
| return ERR_PTR(-EINVAL); |
| |
| rpm_msg = get_msg_from_pool(rc); |
| if (!rpm_msg) |
| return ERR_PTR(-ENOMEM); |
| |
| memcpy(rpm_msg->cmd, cmd, n * sizeof(*cmd)); |
| |
| rpm_msg->msg.state = state; |
| rpm_msg->msg.payload = rpm_msg->cmd; |
| rpm_msg->msg.num_payload = n; |
| |
| return rpm_msg; |
| } |
| |
| /** |
| * rpmh_write_async: Write a batch of RPMH commands |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @state: Active/sleep set |
| * @cmd: The payload data |
| * @n: The number of elements in payload |
| * |
| * Write a batch of RPMH commands, the order of commands is maintained |
| * and will be sent as a single shot. By default the entire set of commands |
| * are considered active only (i.e, will not be cached in wake set, unless |
| * all of them have their corresponding sleep requests). |
| */ |
| int rpmh_write_async(struct rpmh_client *rc, enum rpmh_state state, |
| struct tcs_cmd *cmd, int n) |
| { |
| struct rpmh_msg *rpm_msg; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm_msg = __get_rpmh_msg_async(rc, state, cmd, n); |
| if (IS_ERR(rpm_msg)) |
| return PTR_ERR(rpm_msg); |
| |
| return __rpmh_write(rc, state, rpm_msg); |
| } |
| EXPORT_SYMBOL(rpmh_write_async); |
| |
| /** |
| * rpmh_write: Write a batch of RPMH commands |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @state: Active/sleep set |
| * @cmd: The payload data |
| * @n: The number of elements in payload |
| * |
| * Write a batch of RPMH commands, the order of commands is maintained |
| * and will be sent as a single shot. By default the entire set of commands |
| * are considered active only (i.e, will not be cached in wake set, unless |
| * all of them have their corresponding sleep requests). All requests are |
| * sent as slow path requests. |
| * |
| * May sleep. Do not call from atomic contexts. |
| */ |
| int rpmh_write(struct rpmh_client *rc, enum rpmh_state state, |
| struct tcs_cmd *cmd, int n) |
| { |
| DECLARE_WAIT_QUEUE_HEAD_ONSTACK(waitq); |
| atomic_t wait_count = ATOMIC_INIT(1); |
| DEFINE_RPMH_MSG_ONSTACK(rc, state, &waitq, &wait_count, rpm_msg); |
| int ret; |
| |
| if (IS_ERR_OR_NULL(rc) || !cmd || n <= 0 || n > MAX_RPMH_PAYLOAD) |
| return -EINVAL; |
| |
| might_sleep(); |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| memcpy(rpm_msg.cmd, cmd, n * sizeof(*cmd)); |
| rpm_msg.msg.num_payload = n; |
| |
| ret = __rpmh_write(rc, state, &rpm_msg); |
| if (ret) |
| return ret; |
| |
| wait_event(waitq, atomic_read(&wait_count) == 0); |
| |
| return rpm_msg.err; |
| } |
| EXPORT_SYMBOL(rpmh_write); |
| |
| /** |
| * rpmh_write_passthru: Write multiple batches of RPMH commands without caching |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @state: Active/sleep set |
| * @cmd: The payload data |
| * @n: The array of count of elements in each batch, 0 terminated. |
| * |
| * Write a request to the mailbox controller without caching. If the request |
| * state is ACTIVE or AWAKE, then the requests are treated as completion request |
| * and sent to the controller immediately. The function waits until all the |
| * commands are complete. If the request was to SLEEP or WAKE_ONLY, then the |
| * request is sent as fire-n-forget and no ack is expected. |
| * |
| * May sleep. Do not call from atomic contexts for ACTIVE_ONLY requests. |
| */ |
| int rpmh_write_passthru(struct rpmh_client *rc, enum rpmh_state state, |
| struct tcs_cmd *cmd, int *n) |
| { |
| struct rpmh_msg *rpm_msg[RPMH_MAX_REQ_IN_BATCH]; |
| DECLARE_WAIT_QUEUE_HEAD_ONSTACK(waitq); |
| atomic_t wait_count = ATOMIC_INIT(0); /* overwritten */ |
| int count = 0; |
| int ret, i, j, k; |
| bool complete_set; |
| unsigned long flags; |
| struct rpmh_mbox *rpm; |
| |
| if (IS_ERR_OR_NULL(rc) || !cmd || !n) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| /* Do not allow setting wake votes when in solver mode */ |
| rpm = rc->rpmh; |
| spin_lock_irqsave(&rpm->lock, flags); |
| if (rpm->in_solver_mode && state == RPMH_WAKE_ONLY_STATE) { |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| return -EIO; |
| } |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| while (n[count++]) |
| ; |
| count--; |
| if (!count || count > RPMH_MAX_REQ_IN_BATCH) |
| return -EINVAL; |
| |
| if (state == RPMH_ACTIVE_ONLY_STATE || state == RPMH_AWAKE_STATE) { |
| /* |
| * Ensure the 'complete' bit is set for atleast one command in |
| * each set for active/awake requests. |
| */ |
| for (i = 0, k = 0; i < count; i++, k += n[i]) { |
| complete_set = false; |
| for (j = 0; j < n[i]; j++) { |
| if (cmd[k + j].complete) { |
| complete_set = true; |
| break; |
| } |
| } |
| if (!complete_set) { |
| dev_err(rc->dev, "No completion set for batch"); |
| return -EINVAL; |
| } |
| } |
| } |
| |
| /* Create async request batches */ |
| for (i = 0; i < count; i++) { |
| rpm_msg[i] = __get_rpmh_msg_async(rc, state, cmd, n[i]); |
| if (IS_ERR_OR_NULL(rpm_msg[i])) { |
| /* Clean up our call by spoofing tx_done */ |
| for (j = 0 ; j < i; j++) |
| rpmh_tx_done(&rc->client, &rpm_msg[j]->msg, 0); |
| return PTR_ERR(rpm_msg[i]); |
| } |
| cmd += n[i]; |
| } |
| |
| /* Send if Active or Awake and wait for the whole set to complete */ |
| if (state == RPMH_ACTIVE_ONLY_STATE || state == RPMH_AWAKE_STATE) { |
| might_sleep(); |
| atomic_set(&wait_count, count); |
| for (i = 0; i < count; i++) { |
| rpm_msg[i]->waitq = &waitq; |
| rpm_msg[i]->wait_count = &wait_count; |
| /* Bypass caching and write to mailbox directly */ |
| ret = mbox_send_message(rc->chan, &rpm_msg[i]->msg); |
| if (ret < 0) { |
| pr_err("Error(%d) sending RPM message addr=0x%x\n", |
| ret, rpm_msg[i]->msg.payload[0].addr); |
| break; |
| } |
| } |
| wait_event(waitq, atomic_read(&wait_count) == (count - i)); |
| } else { |
| /* Send Sleep requests to the controller, expect no response */ |
| for (i = 0; i < count; i++) { |
| rpm_msg[i]->waitq = NULL; |
| ret = mbox_send_controller_data(rc->chan, |
| &rpm_msg[i]->msg); |
| /* Clean up our call by spoofing tx_done */ |
| rpmh_tx_done(&rc->client, &rpm_msg[i]->msg, ret); |
| } |
| return 0; |
| } |
| |
| return 0; |
| } |
| EXPORT_SYMBOL(rpmh_write_passthru); |
| |
| /** |
| * rpmh_mode_solver_set: Indicate that the RSC controller hardware has |
| * been configured to be in solver mode |
| * |
| * @rc: The RPMH handle |
| * @enable: Boolean value indicating if the controller is in solver mode. |
| * |
| * When solver mode is enabled, passthru API will not be able to send wake |
| * votes, just awake and active votes. |
| */ |
| int rpmh_mode_solver_set(struct rpmh_client *rc, bool enable) |
| { |
| struct rpmh_mbox *rpm; |
| unsigned long flags; |
| |
| if (IS_ERR_OR_NULL(rc)) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm = rc->rpmh; |
| do { |
| spin_lock_irqsave(&rpm->lock, flags); |
| if (mbox_controller_is_idle(rc->chan)) { |
| rpm->in_solver_mode = enable; |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| break; |
| } |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| udelay(10); |
| } while (1); |
| |
| return 0; |
| } |
| EXPORT_SYMBOL(rpmh_mode_solver_set); |
| |
| /** |
| * rpmh_write_control: Write async control commands to the controller |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @cmd: The payload data |
| * @n: The number of elements in payload |
| * |
| * Write control commands to the controller. The messages are always sent |
| * async. |
| * |
| * May be called from atomic contexts. |
| */ |
| int rpmh_write_control(struct rpmh_client *rc, struct tcs_cmd *cmd, int n) |
| { |
| DEFINE_RPMH_MSG_ONSTACK(rc, 0, NULL, NULL, rpm_msg); |
| |
| if (IS_ERR_OR_NULL(rc) || n > MAX_RPMH_PAYLOAD) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| memcpy(rpm_msg.cmd, cmd, n * sizeof(*cmd)); |
| rpm_msg.msg.num_payload = n; |
| rpm_msg.msg.is_control = true; |
| rpm_msg.msg.is_complete = false; |
| |
| return mbox_send_controller_data(rc->chan, &rpm_msg.msg); |
| } |
| EXPORT_SYMBOL(rpmh_write_control); |
| |
| /** |
| * rpmh_invalidate: Invalidate all sleep and active sets |
| * sets. |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * |
| * Invalidate the sleep and active values in the TCS blocks. |
| * Nothing to do here. |
| */ |
| int rpmh_invalidate(struct rpmh_client *rc) |
| { |
| DEFINE_RPMH_MSG_ONSTACK(rc, 0, NULL, NULL, rpm_msg); |
| struct rpmh_mbox *rpm; |
| unsigned long flags; |
| |
| if (IS_ERR_OR_NULL(rc)) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm = rc->rpmh; |
| rpm_msg.msg.invalidate = true; |
| rpm_msg.msg.is_complete = false; |
| |
| spin_lock_irqsave(&rpm->lock, flags); |
| rpm->dirty = true; |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| return mbox_send_controller_data(rc->chan, &rpm_msg.msg); |
| } |
| EXPORT_SYMBOL(rpmh_invalidate); |
| |
| /** |
| * rpmh_read: Read a resource value |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * @addr: The ePCB address |
| * @resp: The store for the response received from RPMH |
| * |
| * Read a resource value from RPMH. |
| */ |
| int rpmh_read(struct rpmh_client *rc, u32 addr, u32 *resp) |
| { |
| int ret; |
| DECLARE_WAIT_QUEUE_HEAD_ONSTACK(waitq); |
| atomic_t wait_count = ATOMIC_INIT(2); /* wait for rx_cb and tx_done */ |
| DEFINE_RPMH_MSG_ONSTACK(rc, RPMH_ACTIVE_ONLY_STATE, |
| &waitq, &wait_count, rpm_msg); |
| |
| if (IS_ERR_OR_NULL(rc) || !resp) |
| return -EINVAL; |
| |
| might_sleep(); |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| rpm_msg.cmd[0].addr = addr; |
| rpm_msg.cmd[0].data = 0; |
| rpm_msg.msg.num_payload = 1; |
| |
| rpm_msg.msg.is_read = true; |
| |
| ret = mbox_send_message(rc->chan, &rpm_msg.msg); |
| if (ret < 0) |
| return ret; |
| |
| /* Wait until the response is received from RPMH */ |
| wait_event(waitq, atomic_read(&wait_count) == 0); |
| |
| /* Read the data back from the tcs_mbox_msg structrure */ |
| *resp = rpm_msg.cmd[0].data; |
| |
| return rpm_msg.err; |
| } |
| EXPORT_SYMBOL(rpmh_read); |
| |
| static inline int is_req_valid(struct rpmh_req *req) |
| { |
| return (req->sleep_val != UINT_MAX && req->wake_val != UINT_MAX |
| && req->sleep_val != req->wake_val); |
| } |
| |
| int send_single(struct rpmh_client *rc, enum rpmh_state state, u32 addr, |
| u32 data) |
| { |
| DEFINE_RPMH_MSG_ONSTACK(rc, state, NULL, NULL, rpm_msg); |
| |
| /* Wake sets are always complete and sleep sets are not */ |
| rpm_msg.msg.is_complete = (state == RPMH_WAKE_ONLY_STATE); |
| rpm_msg.cmd[0].addr = addr; |
| rpm_msg.cmd[0].data = data; |
| rpm_msg.msg.num_payload = 1; |
| rpm_msg.msg.is_complete = false; |
| |
| return mbox_send_controller_data(rc->chan, &rpm_msg.msg); |
| } |
| |
| /** |
| * rpmh_flush: Flushes the buffered active and sleep sets to TCS |
| * |
| * @rc: The RPMh handle got from rpmh_get_dev_channel |
| * |
| * This function is generally called from the sleep code from the last CPU |
| * that is powering down the entire system. |
| * |
| * Returns -EBUSY if the controller is busy, probably waiting on a response |
| * to a RPMH request sent earlier. |
| */ |
| int rpmh_flush(struct rpmh_client *rc) |
| { |
| struct rpmh_req *p; |
| struct rpmh_mbox *rpm = rc->rpmh; |
| int ret; |
| unsigned long flags; |
| |
| if (IS_ERR_OR_NULL(rc)) |
| return -EINVAL; |
| |
| if (rpmh_standalone) |
| return 0; |
| |
| if (!mbox_controller_is_idle(rc->chan)) |
| return -EBUSY; |
| |
| spin_lock_irqsave(&rpm->lock, flags); |
| if (!rpm->dirty) { |
| pr_debug("Skipping flush, TCS has latest data.\n"); |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| return 0; |
| } |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| /* |
| * Nobody else should be calling this function other than sleep, |
| * hence we can run without locks. |
| */ |
| list_for_each_entry(p, &rc->rpmh->resources, list) { |
| if (!is_req_valid(p)) { |
| pr_debug("%s: skipping RPMH req: a:0x%x s:0x%x w:0x%x", |
| __func__, p->addr, p->sleep_val, p->wake_val); |
| continue; |
| } |
| ret = send_single(rc, RPMH_SLEEP_STATE, p->addr, p->sleep_val); |
| if (ret) |
| return ret; |
| ret = send_single(rc, RPMH_WAKE_ONLY_STATE, p->addr, |
| p->wake_val); |
| if (ret) |
| return ret; |
| } |
| |
| spin_lock_irqsave(&rpm->lock, flags); |
| rpm->dirty = false; |
| spin_unlock_irqrestore(&rpm->lock, flags); |
| |
| return 0; |
| } |
| EXPORT_SYMBOL(rpmh_flush); |
| |
| /** |
| * get_mbox: Get the MBOX controller |
| * @pdev: the platform device |
| * @name: the MBOX name as specified in DT for the device. |
| * @index: the index in the mboxes property if name is not provided. |
| * |
| * Get the MBOX Device node. We will use that to know which |
| * MBOX controller this platform device is intending to talk |
| * to. |
| */ |
| static struct rpmh_mbox *get_mbox(struct platform_device *pdev, |
| const char *name, int index) |
| { |
| int i; |
| struct property *prop; |
| struct of_phandle_args spec; |
| const char *mbox_name; |
| struct rpmh_mbox *rpmh; |
| |
| if (index < 0) { |
| if (!name || !name[0]) |
| return ERR_PTR(-EINVAL); |
| index = 0; |
| of_property_for_each_string(pdev->dev.of_node, |
| "mbox-names", prop, mbox_name) { |
| if (!strcmp(name, mbox_name)) |
| break; |
| index++; |
| } |
| } |
| |
| if (of_parse_phandle_with_args(pdev->dev.of_node, "mboxes", |
| "#mbox-cells", index, &spec)) { |
| dev_dbg(&pdev->dev, "%s: can't parse mboxes property\n", |
| __func__); |
| return ERR_PTR(-ENODEV); |
| } |
| |
| for (i = 0; i < RPMH_MAX_MBOXES; i++) |
| if (mbox_ctrlr[i].mbox_dn == spec.np) { |
| rpmh = &mbox_ctrlr[i]; |
| goto found; |
| } |
| |
| /* A new MBOX */ |
| for (i = 0; i < RPMH_MAX_MBOXES; i++) |
| if (!mbox_ctrlr[i].mbox_dn) |
| break; |
| |
| /* More controllers than expected - not recoverable */ |
| WARN_ON(i == RPMH_MAX_MBOXES); |
| |
| rpmh = &mbox_ctrlr[i]; |
| |
| rpmh->msg_pool = kzalloc(sizeof(struct rpmh_msg) * |
| RPMH_MAX_FAST_RES, GFP_KERNEL); |
| if (!rpmh->msg_pool) |
| return ERR_PTR(-ENOMEM); |
| |
| rpmh->mbox_dn = spec.np; |
| INIT_LIST_HEAD(&rpmh->resources); |
| spin_lock_init(&rpmh->lock); |
| |
| found: |
| of_node_put(spec.np); |
| |
| return rpmh; |
| } |
| |
| static struct rpmh_client *get_rpmh_client(struct platform_device *pdev, |
| const char *name, int index) |
| { |
| struct rpmh_client *rc; |
| int ret = 0; |
| |
| ret = cmd_db_ready(); |
| if (ret) |
| return ERR_PTR(ret); |
| |
| rc = kzalloc(sizeof(*rc), GFP_KERNEL); |
| if (!rc) |
| return ERR_PTR(-ENOMEM); |
| |
| rc->client.rx_callback = rpmh_rx_cb; |
| rc->client.tx_prepare = NULL; |
| rc->client.tx_done = rpmh_tx_done; |
| rc->client.tx_block = false; |
| rc->client.knows_txdone = false; |
| rc->client.dev = &pdev->dev; |
| rc->dev = &pdev->dev; |
| |
| rc->chan = ERR_PTR(-EINVAL); |
| |
| /* Initialize by index or name, whichever is present */ |
| if (index >= 0) |
| rc->chan = mbox_request_channel(&rc->client, index); |
| else if (name) |
| rc->chan = mbox_request_channel_byname(&rc->client, name); |
| |
| if (IS_ERR_OR_NULL(rc->chan)) { |
| ret = PTR_ERR(rc->chan); |
| goto cleanup; |
| } |
| |
| mutex_lock(&rpmh_mbox_mutex); |
| rc->rpmh = get_mbox(pdev, name, index); |
| rpmh_standalone = (cmd_db_is_standalone() > 0); |
| mutex_unlock(&rpmh_mbox_mutex); |
| |
| if (IS_ERR(rc->rpmh)) { |
| ret = PTR_ERR(rc->rpmh); |
| mbox_free_channel(rc->chan); |
| goto cleanup; |
| } |
| |
| return rc; |
| |
| cleanup: |
| kfree(rc); |
| return ERR_PTR(ret); |
| } |
| |
| /** |
| * rpmh_get_byname: Get the RPMh handle by mbox name |
| * |
| * @pdev: the platform device which needs to communicate with RPM |
| * accelerators |
| * @name: The mbox-name assigned to the client's mailbox handle |
| * |
| * May sleep. |
| */ |
| struct rpmh_client *rpmh_get_byname(struct platform_device *pdev, |
| const char *name) |
| { |
| return get_rpmh_client(pdev, name, -1); |
| } |
| EXPORT_SYMBOL(rpmh_get_byname); |
| |
| /** |
| * rpmh_get_byindex: Get the RPMh handle by mbox index |
| * |
| * @pdev: the platform device which needs to communicate with RPM |
| * accelerators |
| * @index : The index of the mbox tuple as specified in order in DT |
| * |
| * May sleep. |
| */ |
| struct rpmh_client *rpmh_get_byindex(struct platform_device *pdev, |
| int index) |
| { |
| return get_rpmh_client(pdev, NULL, index); |
| } |
| EXPORT_SYMBOL(rpmh_get_byindex); |
| |
| /** |
| * rpmh_release: Release the RPMH client |
| * |
| * @rc: The RPMh handle to be freed. |
| */ |
| void rpmh_release(struct rpmh_client *rc) |
| { |
| if (rc && !IS_ERR_OR_NULL(rc->chan)) |
| mbox_free_channel(rc->chan); |
| |
| kfree(rc); |
| } |
| EXPORT_SYMBOL(rpmh_release); |