diff options
Diffstat (limited to 'gxp-hw-mailbox-driver.c')
-rw-r--r-- | gxp-hw-mailbox-driver.c | 323 |
1 files changed, 0 insertions, 323 deletions
diff --git a/gxp-hw-mailbox-driver.c b/gxp-hw-mailbox-driver.c deleted file mode 100644 index 8430a65..0000000 --- a/gxp-hw-mailbox-driver.c +++ /dev/null @@ -1,323 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -/* - * GXP hardware-based mailbox driver implementation. - * - * Copyright (C) 2021 Google LLC - */ - -#include <asm/barrier.h> -#include <linux/bitops.h> -#include <linux/interrupt.h> -#include <linux/kthread.h> -#include <linux/of_irq.h> -#include <linux/spinlock.h> - -#include "gxp-mailbox-driver.h" -#include "gxp-mailbox-regs.h" -#include "gxp-mailbox.h" - -static u32 csr_read(struct gxp_mailbox *mailbox, uint reg_offset) -{ - return readl(mailbox->csr_reg_base + reg_offset); -} - -static void csr_write(struct gxp_mailbox *mailbox, uint reg_offset, u32 value) -{ - writel(value, mailbox->csr_reg_base + reg_offset); -} - -static u32 data_read(struct gxp_mailbox *mailbox, uint reg_offset) -{ - return readl(mailbox->data_reg_base + reg_offset); -} - -static void data_write(struct gxp_mailbox *mailbox, uint reg_offset, - u32 value) -{ - writel(value, mailbox->data_reg_base + reg_offset); -} - -/* IRQ Handling */ - -/* Interrupt to signal a response from the device to host */ -#define MBOX_DEVICE_TO_HOST_RESPONSE_IRQ_MASK BIT(0) - -static irqreturn_t mailbox_irq_handler(int irq, void *arg) -{ - u32 masked_status; - struct gxp_mailbox *mailbox = (struct gxp_mailbox *) arg; - struct work_struct **handlers = mailbox->interrupt_handlers; - u32 next_int; - - /* Contains only the non-masked, pending interrupt bits */ - masked_status = gxp_mailbox_get_host_mask_status(mailbox); - - /* Clear all pending IRQ bits */ - gxp_mailbox_clear_host_interrupt(mailbox, masked_status); - - if (masked_status & MBOX_DEVICE_TO_HOST_RESPONSE_IRQ_MASK) { - mailbox->handle_irq(mailbox); - masked_status &= ~MBOX_DEVICE_TO_HOST_RESPONSE_IRQ_MASK; - } - - while ((next_int = ffs(masked_status))) { - next_int--; /* ffs returns 1-based indices */ - masked_status &= ~BIT(next_int); - - if (handlers[next_int]) - schedule_work(handlers[next_int]); - else - pr_err_ratelimited( - "mailbox%d: received unknown interrupt bit 0x%X\n", - mailbox->core_id, next_int); - } - - return IRQ_HANDLED; -} - -static void register_irq(struct gxp_mailbox *mailbox) -{ - int err; - unsigned int virq; - - virq = irq_of_parse_and_map(mailbox->gxp->dev->of_node, - mailbox->core_id); - if (!virq) { - pr_err("Unable to parse interrupt for core %d from the DT\n", - mailbox->core_id); - return; - } - - err = request_irq(virq, mailbox_irq_handler, /*flags=*/ 0, - "aurora_mbx_irq", (void *) mailbox); - if (err) { - pr_err("Unable to register IRQ num=%d; error=%d\n", virq, err); - return; - } - - mailbox->interrupt_virq = virq; - pr_debug("Core %d's mailbox interrupt registered as IRQ %u.\n", - mailbox->core_id, virq); -} - -static void unregister_irq(struct gxp_mailbox *mailbox) -{ - if (mailbox->interrupt_virq) { - pr_debug("Freeing IRQ %d\n", mailbox->interrupt_virq); - free_irq(mailbox->interrupt_virq, mailbox); - mailbox->interrupt_virq = 0; - } -} - -/* gxp-mailbox-driver.h interface */ - -void gxp_mailbox_driver_init(struct gxp_mailbox *mailbox) -{ - spin_lock_init(&mailbox->cmd_tail_resp_head_lock); - spin_lock_init(&mailbox->cmd_head_resp_tail_lock); -} - -void gxp_mailbox_driver_exit(struct gxp_mailbox *mailbox) -{ - /* Nothing to cleanup */ -} - -void gxp_mailbox_driver_enable_interrupts(struct gxp_mailbox *mailbox) -{ - register_irq(mailbox); -} - -void gxp_mailbox_driver_disable_interrupts(struct gxp_mailbox *mailbox) -{ - unregister_irq(mailbox); -} - -void __iomem *gxp_mailbox_get_csr_base(struct gxp_dev *gxp, uint index) -{ - return gxp->mbx[index].vaddr; -} - -void __iomem *gxp_mailbox_get_data_base(struct gxp_dev *gxp, uint index) -{ - return gxp->mbx[index].vaddr + 0x80; -} - -/* gxp-mailbox-driver.h: CSR-based calls */ - -void gxp_mailbox_reset_hw(struct gxp_mailbox *mailbox) -{ - csr_write(mailbox, MBOX_MCUCTLR_OFFSET, 1); -} - -void gxp_mailbox_generate_device_interrupt(struct gxp_mailbox *mailbox, - u32 int_mask) -{ - /* - * Ensure all memory writes have been committed to memory before - * signalling to the device to read from them. This avoids the scenario - * where the interrupt trigger write gets delivered to the MBX HW before - * the DRAM transactions made it to DRAM since they're Normal - * transactions and can be re-ordered and backed off behind other - * transfers. - */ - wmb(); - - csr_write(mailbox, MBOX_INTGR0_OFFSET, int_mask); -} - -u32 gxp_mailbox_get_device_mask_status(struct gxp_mailbox *mailbox) -{ - return csr_read(mailbox, MBOX_INTMSR0_OFFSET); -} - -void gxp_mailbox_clear_host_interrupt(struct gxp_mailbox *mailbox, u32 int_mask) -{ - csr_write(mailbox, MBOX_INTCR1_OFFSET, int_mask); -} - -void gxp_mailbox_mask_host_interrupt(struct gxp_mailbox *mailbox, u32 int_mask) -{ - csr_write(mailbox, MBOX_INTMR1_OFFSET, int_mask); -} - -u32 gxp_mailbox_get_host_mask_status(struct gxp_mailbox *mailbox) -{ - return csr_read(mailbox, MBOX_INTMSR1_OFFSET); -} - -/* gxp-mailbox-driver.h: Data register-based calls */ - -void gxp_mailbox_write_status(struct gxp_mailbox *mailbox, u32 status) -{ - data_write(mailbox, MBOX_STATUS_OFFSET, status); -} - -void gxp_mailbox_write_descriptor(struct gxp_mailbox *mailbox, - dma_addr_t descriptor_addr) -{ - data_write(mailbox, MBOX_DESCRIPTOR_ADDR_OFFSET, (u32)descriptor_addr); -} - -void gxp_mailbox_write_cmd_queue_tail(struct gxp_mailbox *mailbox, u16 val) -{ - u32 current_resp_head; - u32 new_cmd_tail; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_tail_resp_head_lock, flags); - - current_resp_head = data_read(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET) & - RESP_HEAD_MASK; - new_cmd_tail = (u32)val << CMD_TAIL_SHIFT; - data_write(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET, - new_cmd_tail | current_resp_head); - - spin_unlock_irqrestore(&mailbox->cmd_tail_resp_head_lock, flags); -} - -void gxp_mailbox_write_resp_queue_head(struct gxp_mailbox *mailbox, u16 val) -{ - u32 current_cmd_tail; - u32 new_resp_head; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_tail_resp_head_lock, flags); - - current_cmd_tail = data_read(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET) & - CMD_TAIL_MASK; - new_resp_head = (u32)val << RESP_HEAD_SHIFT; - data_write(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET, - current_cmd_tail | new_resp_head); - - spin_unlock_irqrestore(&mailbox->cmd_tail_resp_head_lock, flags); -} - -u16 gxp_mailbox_read_cmd_queue_head(struct gxp_mailbox *mailbox) -{ - u32 reg_val; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_head_resp_tail_lock, flags); - - reg_val = data_read(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET); - - spin_unlock_irqrestore(&mailbox->cmd_head_resp_tail_lock, flags); - - return (u16)((reg_val & CMD_HEAD_MASK) >> CMD_HEAD_SHIFT); -} - -u16 gxp_mailbox_read_resp_queue_tail(struct gxp_mailbox *mailbox) -{ - u32 reg_val; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_head_resp_tail_lock, flags); - - reg_val = data_read(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET); - - spin_unlock_irqrestore(&mailbox->cmd_head_resp_tail_lock, flags); - - return (u16)((reg_val & RESP_TAIL_MASK) >> RESP_TAIL_SHIFT); -} - -void gxp_mailbox_write_cmd_queue_head(struct gxp_mailbox *mailbox, u16 val) -{ - u32 current_resp_tail; - u32 new_cmd_head; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_head_resp_tail_lock, flags); - - current_resp_tail = data_read(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET) & - RESP_TAIL_MASK; - new_cmd_head = (u32)val << CMD_HEAD_SHIFT; - data_write(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET, - new_cmd_head | current_resp_tail); - - spin_unlock_irqrestore(&mailbox->cmd_head_resp_tail_lock, flags); -} - -void gxp_mailbox_write_resp_queue_tail(struct gxp_mailbox *mailbox, u16 val) -{ - u32 current_cmd_head; - u32 new_resp_tail; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_head_resp_tail_lock, flags); - - current_cmd_head = data_read(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET) & - CMD_HEAD_MASK; - new_resp_tail = (u32)val << RESP_TAIL_SHIFT; - data_write(mailbox, MBOX_CMD_HEAD_RESP_TAIL_OFFSET, - current_cmd_head | new_resp_tail); - - spin_unlock_irqrestore(&mailbox->cmd_head_resp_tail_lock, flags); -} - -u16 gxp_mailbox_read_cmd_queue_tail(struct gxp_mailbox *mailbox) -{ - u32 reg_val; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_tail_resp_head_lock, flags); - - reg_val = data_read(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET); - - spin_unlock_irqrestore(&mailbox->cmd_tail_resp_head_lock, flags); - - return (u16)((reg_val & CMD_TAIL_MASK) >> CMD_TAIL_SHIFT); -} - -u16 gxp_mailbox_read_resp_queue_head(struct gxp_mailbox *mailbox) -{ - u32 reg_val; - unsigned long flags; - - spin_lock_irqsave(&mailbox->cmd_tail_resp_head_lock, flags); - - reg_val = data_read(mailbox, MBOX_CMD_TAIL_RESP_HEAD_OFFSET); - - spin_unlock_irqrestore(&mailbox->cmd_tail_resp_head_lock, flags); - - return (u16)((reg_val & RESP_HEAD_MASK) >> RESP_HEAD_SHIFT); -} |