123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292 |
- /**
- * Copyright (c) 1997-2015, Actions Semi Co., Inc.
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <kernel.h>
- #include <device.h>
- #include <drivers/dsp.h>
- #include <soc_dsp.h>
- #include "dsp_inner.h"
- #include <init.h>
- #include "os_common_api.h"
- #define CONFIG_DSP_WORK_Q_STACK_SIZE (2048)
- #define CONFIG_DSP_WORK_Q_PRIORITY (-2)
- static struct k_work_q dsp_workq;
- static K_KERNEL_STACK_DEFINE(dsp_workq_stack, CONFIG_DSP_WORK_Q_STACK_SIZE);
- struct dsp_message pagemiss_message;
- struct device *pagemiss_dev = NULL;
- int dsp_acts_register_message_handler(struct device *dev, dsp_message_handler handler)
- {
- struct dsp_acts_data *dsp_data = dev->data;
- unsigned int irq_key;
- irq_key = irq_lock();
- dsp_data->msg_handler = handler;
- irq_unlock(irq_key);
- return 0;
- }
- int dsp_acts_unregister_message_handler(struct device *dev)
- {
- struct dsp_acts_data *dsp_data = dev->data;
- unsigned int irq_key;
- irq_key = irq_lock();
- dsp_data->msg_handler = NULL;
- irq_unlock(irq_key);
- return 0;
- }
- int dsp_acts_send_message_response(struct device *dev, struct dsp_message *message, int res)
- {
- const struct dsp_acts_config *dsp_cfg = dev->config;
- //struct dsp_acts_data *dsp_data = dev->data;
- struct dsp_protocol_mailbox *mailbox = dsp_cfg->cpu_mailbox;
- uint32_t status = MSG_STATUS(mailbox->msg);
- /* FIXME: force set result fail */
- if (res < 0)
- message->result = DSP_FAIL;
- /* confirm message received */
- status &= ~(MSG_FLAG_BUSY | MSG_FLAG_DONE | MSG_FLAG_RPLY | MSG_FLAG_FAIL);
- status |= MSG_FLAG_ACK;
- switch (message->result) {
- case DSP_REPLY:
- /* always copy back result */
- mailbox->param1 = message->param1;
- mailbox->param2 = message->param2;
- status |= MSG_FLAG_RPLY;
- case DSP_DONE:
- status |= MSG_FLAG_DONE;
- break;
- case DSP_FAIL:
- status |= MSG_FLAG_FAIL;
- break;
- default:
- break;
- }
- mailbox->msg = MAILBOX_MSG(message->id, status);
- printk("dsp_acts_send_message_response id %d status %x \n",message->id, status);
- return res;
- }
- static void dsp_page_miss_handle(struct k_work *work)
- {
- int res = dsp_acts_handle_image_pagemiss(pagemiss_dev, pagemiss_message.param1);
- dsp_acts_send_message_response(pagemiss_dev, &pagemiss_message, res);
- }
- K_WORK_DEFINE(dsp_page_miss_work, dsp_page_miss_handle);
- /* This function called in irq context */
- int dsp_acts_recv_message(struct device *dev)
- {
- const struct dsp_acts_config *dsp_cfg = dev->config;
- struct dsp_acts_data *dsp_data = dev->data;
- struct dsp_protocol_mailbox *mailbox = dsp_cfg->cpu_mailbox;
- struct dsp_message message = {
- .id = MSG_ID(mailbox->msg),
- .result = DSP_DONE,
- .owner = mailbox->owner,
- .param1 = mailbox->param1,
- .param2 = mailbox->param2,
- };
- uint32_t status = MSG_STATUS(mailbox->msg);
- int res = 0;
- if (dsp_data->pm_status == DSP_STATUS_POWEROFF){
- printk("%s: deaded\n", __func__);
- return -ENODEV;
- }
- if (!(status & MSG_FLAG_BUSY)) {
- printk("%s: busy flag of msg (%u:%u) not set\n", __func__,
- message.owner, message.id);
- return -EINVAL;
- }
- if (status & MSG_FLAG_ACK) {
- printk("%s: ack of msg (%u:%u) already set\n", __func__,
- message.owner, message.id);
- return -EINVAL;
- }
- #if 0
- printk("msg_id %d param %x %x\n", message.id, message.param1, message.param2);
- #endif
- switch (message.id) {
- case DSP_MSG_REQUEST_BOOTARGS:
- message.param1 = mcu_to_dsp_address((uint32_t)&dsp_data->bootargs, DATA_ADDR);
- message.result = DSP_REPLY;
- break;
- case DSP_MSG_STATE_CHANGED:
- switch (message.param1) {
- case DSP_TASK_STARTED:
- printk("%s: started param2 %d \n", __func__, message.param2);
- if (message.param2 == DSP_NEED_SYNC_CLOCK) {
- dsp_data->need_sync_clock = 'C';
- message.param1 = DSP_NEED_SYNC_CLOCK;
- message.param2 = 0;
- message.result = DSP_REPLY;
- } else {
- dsp_data->need_sync_clock = 0;
- message.param1 = 0;
- message.param2 = 0;
- message.result = DSP_REPLY;
- }
- break;
- case DSP_TASK_SUSPENDED:
- printk("%s: suspended\n", __func__);
- break;
- case DSP_TASK_RESUMED:
- printk("%s: resumed\n", __func__);
- break;
- default:
- break;
- }
- k_sem_give(&dsp_data->msg_sem);
- break;
- case DSP_MSG_PAGE_MISS:
- pagemiss_dev = dev;
- memcpy(&pagemiss_message, &message, sizeof(struct dsp_message));
- os_work_submit_to_queue(&dsp_workq, &dsp_page_miss_work);
- return res;
- case DSP_MSG_PAGE_FLUSH:
- res = dsp_acts_handle_image_pageflush(dev, message.param1);
- break;
- case DSP_MSG_NULL:
- break;
- case DSP_MSG_KICK:
- default:
- if (dsp_data->msg_handler) {
- res = dsp_data->msg_handler(&message);
- } else {
- printk("%s: unexpected msg %u\n", __func__, message.id);
- res = -ENOMSG;
- }
- break;
- }
- res = dsp_acts_send_message_response(dev, &message, res);
- return res;
- }
- static int wait_ack_timeout(struct dsp_protocol_mailbox *mailbox, int usec_to_wait)
- {
- do {
- uint32_t status = MSG_STATUS(mailbox->msg);
- if ((status & MSG_FLAG_ACK) || (usec_to_wait-- <= 0))
- break;
- k_busy_wait(1);
- } while (1);
- return usec_to_wait;
- }
- int dsp_acts_send_message(struct device *dev, struct dsp_message *message)
- {
- struct dsp_acts_data *dsp_data = dev->data;
- const struct dsp_acts_config *dsp_cfg = dev->config;
- struct dsp_protocol_mailbox *mailbox = dsp_cfg->dsp_mailbox;
- uint32_t status = MSG_STATUS(mailbox->msg);
- int ret = 0;
- if (dsp_data->pm_status == DSP_STATUS_POWEROFF)
- return -EFAULT;
- if (!(status & MSG_FLAG_ACK)) {
- printk("%s: ack of msg (%u:%u) not yet set by dsp\n", __func__,
- mailbox->owner, MSG_ID(mailbox->msg));
- return -EBUSY;
- }
- if (status & MSG_FLAG_BUSY) {
- printk("%s: busy flag of msg (%u:%u) not yet cleared by dsp\n",
- __func__, mailbox->owner, MSG_ID(mailbox->msg));
- return -EBUSY;
- }
- if (k_is_in_isr()) {
- printk("%s: send msg (%u:%u) in isr\n", __func__,
- message->owner, message->id);
- } else {
- k_mutex_lock(&dsp_data->msg_mutex, K_FOREVER);
- }
- mailbox->msg = MAILBOX_MSG(message->id, MSG_FLAG_BUSY);
- mailbox->owner = message->owner;
- mailbox->param1 = message->param1;
- mailbox->param2 = message->param2;
- /* trigger irq to dsp */
- mcu_trigger_irq_to_dsp();
- /* Wait for the ack bit: 1ms timeout (1us * 1000) */
- wait_ack_timeout(mailbox, 1000);
- /* de-trigger irq to dsp */
- mcu_untrigger_irq_to_dsp();
- status = MSG_STATUS(mailbox->msg);
- if (!(status & MSG_FLAG_ACK)) {
- printk("%s: ack of msg %u wait timeout\n", __func__, message->id);
- message->result = DSP_NOACK;
- ret = -ETIMEDOUT;
- goto EXIT;
- } else if (status & MSG_FLAG_FAIL) {
- message->result = DSP_FAIL;
- ret = -ENOTSUP;
- goto EXIT;
- } else if (status & MSG_FLAG_RPLY) {
- message->param1 = mailbox->param1;
- message->param2 = mailbox->param2;
- message->result = DSP_REPLY;
- } else if (status & MSG_FLAG_DONE) {
- message->result = DSP_DONE;
- } else {
- message->result = DSP_INPROGRESS;
- }
- EXIT:
- if (!k_is_in_isr())
- k_mutex_unlock(&dsp_data->msg_mutex);
- return ret;
- }
- static int dsp_work_q_init(const struct device *dev)
- {
- ARG_UNUSED(dev);
- struct k_work_queue_config cfg = {
- .name = "dspworkq",
- .no_yield = false,
- };
- k_work_queue_start(&dsp_workq,
- dsp_workq_stack,
- K_KERNEL_STACK_SIZEOF(dsp_workq_stack),
- CONFIG_DSP_WORK_Q_PRIORITY, &cfg);
- return 0;
- }
- SYS_INIT(dsp_work_q_init, POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|