/* * Mailbox internal functions * * Copyright (C) 2006 Nokia Corporation * Written by: Hiroshi DOYU * * This file is subject to the terms and conditions of the GNU General Public * License. See the file "COPYING" in the main directory of this archive * for more details. */ #ifndef __ARCH_ARM_PLAT_MAILBOX_H #define __ARCH_ARM_PLAT_MAILBOX_H /* * Mailbox queue handling API */ #define MBQ_DEPTH 16 struct omap_mbq { rwlock_t lock; mbox_msg_t msg[MBQ_DEPTH]; mbox_msg_t *rp, *wp; }; static inline int mbq_init(struct omap_mbq **addr) { struct omap_mbq *m = kmalloc(sizeof(struct omap_mbq), GFP_KERNEL); if (!m) return -ENOMEM; rwlock_init(&m->lock); write_lock_irq(&m->lock); m->rp = m->wp = &m->msg[0]; write_unlock_irq(&m->lock); *addr = m; return 0; } static inline int mbq_empty(struct omap_mbq *mbq) { int ret; read_lock_irq(&mbq->lock); ret = (mbq->rp == mbq->wp); read_unlock_irq(&mbq->lock); return ret; } static inline int mbq_full(struct omap_mbq *mbq) { int ret; mbox_msg_t *p; read_lock_irq(&mbq->lock); p = mbq->wp; if (++p == &mbq->msg[MBQ_DEPTH]) p = &mbq->msg[0]; ret = (p == mbq->rp); read_unlock_irq(&mbq->lock); return ret; } static inline int mbq_add(struct omap_mbq *mbq, mbox_msg_t msg) { int ret = 0; write_lock_irq(&mbq->lock); *mbq->wp = msg; if (++mbq->wp == &mbq->msg[MBQ_DEPTH]) mbq->wp = &mbq->msg[0]; if (mbq->wp == mbq->rp) /* full */ ret = -1;; write_unlock_irq(&mbq->lock); return ret; } static inline mbox_msg_t mbq_get(struct omap_mbq *mbq) { mbox_msg_t msg; write_lock_irq(&mbq->lock); msg = *mbq->rp; if (++mbq->rp == &mbq->msg[MBQ_DEPTH]) mbq->rp = &mbq->msg[0]; write_unlock_irq(&mbq->lock); return msg; } static inline void mbq_exit(struct omap_mbq **addr) { if (*addr) kfree(*addr); } /* * Mailbox sequence bit API */ #if defined(CONFIG_ARCH_OMAP1) # define MBOX_USE_SEQ_BIT #elif defined(CONFIG_ARCH_OMAP2) # define MBOX_USE_SEQ_BIT #endif #ifdef MBOX_USE_SEQ_BIT /* seq_rcv should be initialized with any value other than * 0 and 1 << 31, to allow either value for the first * message. */ static inline void mbox_seq_init(struct omap_mbox *mbox) { /* any value other than 0 and 1 << 31 */ mbox->seq_rcv = 0xffffffff; } static inline void mbox_seq_toggle(struct omap_mbox *mbox, mbox_msg_t * msg) { /* add seq_snd to msg */ *msg = (*msg & 0x7fffffff) | mbox->seq_snd; /* flip seq_snd */ mbox->seq_snd ^= 1 << 31; } static inline int mbox_seq_test(struct omap_mbox *mbox, mbox_msg_t msg) { mbox_msg_t seq = msg & (1 << 31); if (seq == mbox->seq_rcv) return -1; mbox->seq_rcv = seq; return 0; } #else static inline void mbox_seq_init(struct omap_mbox *mbox) { } static inline void mbox_seq_toggle(struct omap_mbox *mbox, mbox_msg_t * msg) { } static inline int mbox_seq_test(struct omap_mbox *mbox, mbox_msg_t msg) { return 0; } #endif /* Mailbox FIFO handle functions */ static inline mbox_msg_t mbox_fifo_read(struct omap_mbox *mbox) { return mbox->ops->fifo_read(mbox); } static inline void mbox_fifo_write(struct omap_mbox *mbox, mbox_msg_t msg) { mbox->ops->fifo_write(mbox, msg); } static inline int mbox_fifo_empty(struct omap_mbox *mbox) { return mbox->ops->fifo_empty(mbox); } static inline int mbox_fifo_full(struct omap_mbox *mbox) { return mbox->ops->fifo_full(mbox); } /* Mailbox IRQ handle functions */ static inline void enable_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq) { mbox->ops->enable_irq(mbox, irq); } static inline void disable_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq) { mbox->ops->disable_irq(mbox, irq); } static inline void ack_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq) { if (mbox->ops->ack_irq) mbox->ops->ack_irq(mbox, irq); } static inline int is_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq) { return mbox->ops->is_irq(mbox, irq); } #endif /* __ARCH_ARM_PLAT_MAILBOX_H */