123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387 |
- /*
- * include/asm-xtensa/variant-s6000/dmac.h
- *
- * 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.
- *
- * Copyright (C) 2006 Tensilica Inc.
- * Copyright (C) 2008 Emlix GmbH <info@emlix.com>
- * Authors: Fabian Godehardt <fg@emlix.com>
- * Oskar Schirmer <os@emlix.com>
- * Daniel Gloeckner <dg@emlix.com>
- */
- #ifndef __ASM_XTENSA_S6000_DMAC_H
- #define __ASM_XTENSA_S6000_DMAC_H
- #include <linux/io.h>
- #include <variant/hardware.h>
- /* DMA global */
- #define S6_DMA_INTSTAT0 0x000
- #define S6_DMA_INTSTAT1 0x004
- #define S6_DMA_INTENABLE0 0x008
- #define S6_DMA_INTENABLE1 0x00C
- #define S6_DMA_INTRAW0 0x010
- #define S6_DMA_INTRAW1 0x014
- #define S6_DMA_INTCLEAR0 0x018
- #define S6_DMA_INTCLEAR1 0x01C
- #define S6_DMA_INTSET0 0x020
- #define S6_DMA_INTSET1 0x024
- #define S6_DMA_INT0_UNDER 0
- #define S6_DMA_INT0_OVER 16
- #define S6_DMA_INT1_CHANNEL 0
- #define S6_DMA_INT1_MASTER 16
- #define S6_DMA_INT1_MASTER_MASK 7
- #define S6_DMA_TERMCNTIRQSTAT 0x028
- #define S6_DMA_TERMCNTIRQCLR 0x02C
- #define S6_DMA_TERMCNTIRQSET 0x030
- #define S6_DMA_PENDCNTIRQSTAT 0x034
- #define S6_DMA_PENDCNTIRQCLR 0x038
- #define S6_DMA_PENDCNTIRQSET 0x03C
- #define S6_DMA_LOWWMRKIRQSTAT 0x040
- #define S6_DMA_LOWWMRKIRQCLR 0x044
- #define S6_DMA_LOWWMRKIRQSET 0x048
- #define S6_DMA_MASTERERRINFO 0x04C
- #define S6_DMA_MASTERERR_CHAN(n) (4*(n))
- #define S6_DMA_MASTERERR_CHAN_MASK 0xF
- #define S6_DMA_DESCRFIFO0 0x050
- #define S6_DMA_DESCRFIFO1 0x054
- #define S6_DMA_DESCRFIFO2 0x058
- #define S6_DMA_DESCRFIFO2_AUTODISABLE 24
- #define S6_DMA_DESCRFIFO3 0x05C
- #define S6_DMA_MASTER0START 0x060
- #define S6_DMA_MASTER0END 0x064
- #define S6_DMA_MASTER1START 0x068
- #define S6_DMA_MASTER1END 0x06C
- #define S6_DMA_NEXTFREE 0x070
- #define S6_DMA_NEXTFREE_CHAN 0
- #define S6_DMA_NEXTFREE_CHAN_MASK 0x1F
- #define S6_DMA_NEXTFREE_ENA 16
- #define S6_DMA_NEXTFREE_ENA_MASK ((1 << 16) - 1)
- #define S6_DMA_DPORTCTRLGRP(p) ((p) * 4 + 0x074)
- #define S6_DMA_DPORTCTRLGRP_FRAMEREP 0
- #define S6_DMA_DPORTCTRLGRP_NRCHANS 1
- #define S6_DMA_DPORTCTRLGRP_NRCHANS_1 0
- #define S6_DMA_DPORTCTRLGRP_NRCHANS_3 1
- #define S6_DMA_DPORTCTRLGRP_NRCHANS_4 2
- #define S6_DMA_DPORTCTRLGRP_NRCHANS_2 3
- #define S6_DMA_DPORTCTRLGRP_ENA 31
- /* DMA per channel */
- #define DMA_CHNL(dmac, n) ((dmac) + 0x1000 + (n) * 0x100)
- #define DMA_INDEX_CHNL(addr) (((addr) >> 8) & 0xF)
- #define DMA_MASK_DMAC(addr) ((addr) & 0xFFFF0000)
- #define S6_DMA_CHNCTRL 0x000
- #define S6_DMA_CHNCTRL_ENABLE 0
- #define S6_DMA_CHNCTRL_PAUSE 1
- #define S6_DMA_CHNCTRL_PRIO 2
- #define S6_DMA_CHNCTRL_PRIO_MASK 3
- #define S6_DMA_CHNCTRL_PERIPHXFER 4
- #define S6_DMA_CHNCTRL_PERIPHENA 5
- #define S6_DMA_CHNCTRL_SRCINC 6
- #define S6_DMA_CHNCTRL_DSTINC 7
- #define S6_DMA_CHNCTRL_BURSTLOG 8
- #define S6_DMA_CHNCTRL_BURSTLOG_MASK 7
- #define S6_DMA_CHNCTRL_DESCFIFODEPTH 12
- #define S6_DMA_CHNCTRL_DESCFIFODEPTH_MASK 0x1F
- #define S6_DMA_CHNCTRL_DESCFIFOFULL 17
- #define S6_DMA_CHNCTRL_BWCONSEL 18
- #define S6_DMA_CHNCTRL_BWCONENA 19
- #define S6_DMA_CHNCTRL_PENDGCNTSTAT 20
- #define S6_DMA_CHNCTRL_PENDGCNTSTAT_MASK 0x3F
- #define S6_DMA_CHNCTRL_LOWWMARK 26
- #define S6_DMA_CHNCTRL_LOWWMARK_MASK 0xF
- #define S6_DMA_CHNCTRL_TSTAMP 30
- #define S6_DMA_TERMCNTNB 0x004
- #define S6_DMA_TERMCNTNB_MASK 0xFFFF
- #define S6_DMA_TERMCNTTMO 0x008
- #define S6_DMA_TERMCNTSTAT 0x00C
- #define S6_DMA_TERMCNTSTAT_MASK 0xFF
- #define S6_DMA_CMONCHUNK 0x010
- #define S6_DMA_SRCSKIP 0x014
- #define S6_DMA_DSTSKIP 0x018
- #define S6_DMA_CUR_SRC 0x024
- #define S6_DMA_CUR_DST 0x028
- #define S6_DMA_TIMESTAMP 0x030
- /* DMA channel lists */
- #define S6_DPDMA_CHAN(stream, channel) (4 * (stream) + (channel))
- #define S6_DPDMA_NB 16
- #define S6_HIFDMA_GMACTX 0
- #define S6_HIFDMA_GMACRX 1
- #define S6_HIFDMA_I2S0 2
- #define S6_HIFDMA_I2S1 3
- #define S6_HIFDMA_EGIB 4
- #define S6_HIFDMA_PCITX 5
- #define S6_HIFDMA_PCIRX 6
- #define S6_HIFDMA_NB 7
- #define S6_NIDMA_NB 4
- #define S6_LMSDMA_NB 12
- /* controller access */
- #define S6_DMAC_NB 4
- #define S6_DMAC_INDEX(dmac) (((unsigned)(dmac) >> 18) % S6_DMAC_NB)
- struct s6dmac_ctrl {
- u32 dmac;
- spinlock_t lock;
- u8 chan_nb;
- };
- extern struct s6dmac_ctrl s6dmac_ctrl[S6_DMAC_NB];
- /* DMA control, per channel */
- static inline int s6dmac_fifo_full(u32 dmac, int chan)
- {
- return (readl(DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL)
- & (1 << S6_DMA_CHNCTRL_DESCFIFOFULL)) && 1;
- }
- static inline int s6dmac_termcnt_irq(u32 dmac, int chan)
- {
- u32 m = 1 << chan;
- int r = (readl(dmac + S6_DMA_TERMCNTIRQSTAT) & m) && 1;
- if (r)
- writel(m, dmac + S6_DMA_TERMCNTIRQCLR);
- return r;
- }
- static inline int s6dmac_pendcnt_irq(u32 dmac, int chan)
- {
- u32 m = 1 << chan;
- int r = (readl(dmac + S6_DMA_PENDCNTIRQSTAT) & m) && 1;
- if (r)
- writel(m, dmac + S6_DMA_PENDCNTIRQCLR);
- return r;
- }
- static inline int s6dmac_lowwmark_irq(u32 dmac, int chan)
- {
- int r = (readl(dmac + S6_DMA_LOWWMRKIRQSTAT) & (1 << chan)) ? 1 : 0;
- if (r)
- writel(1 << chan, dmac + S6_DMA_LOWWMRKIRQCLR);
- return r;
- }
- static inline u32 s6dmac_pending_count(u32 dmac, int chan)
- {
- return (readl(DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL)
- >> S6_DMA_CHNCTRL_PENDGCNTSTAT)
- & S6_DMA_CHNCTRL_PENDGCNTSTAT_MASK;
- }
- static inline void s6dmac_set_terminal_count(u32 dmac, int chan, u32 n)
- {
- n &= S6_DMA_TERMCNTNB_MASK;
- n |= readl(DMA_CHNL(dmac, chan) + S6_DMA_TERMCNTNB)
- & ~S6_DMA_TERMCNTNB_MASK;
- writel(n, DMA_CHNL(dmac, chan) + S6_DMA_TERMCNTNB);
- }
- static inline u32 s6dmac_get_terminal_count(u32 dmac, int chan)
- {
- return (readl(DMA_CHNL(dmac, chan) + S6_DMA_TERMCNTNB))
- & S6_DMA_TERMCNTNB_MASK;
- }
- static inline u32 s6dmac_timestamp(u32 dmac, int chan)
- {
- return readl(DMA_CHNL(dmac, chan) + S6_DMA_TIMESTAMP);
- }
- static inline u32 s6dmac_cur_src(u32 dmac, int chan)
- {
- return readl(DMA_CHNL(dmac, chan) + S6_DMA_CUR_SRC);
- }
- static inline u32 s6dmac_cur_dst(u32 dmac, int chan)
- {
- return readl(DMA_CHNL(dmac, chan) + S6_DMA_CUR_DST);
- }
- static inline void s6dmac_disable_chan(u32 dmac, int chan)
- {
- u32 ctrl;
- writel(readl(DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL)
- & ~(1 << S6_DMA_CHNCTRL_ENABLE),
- DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL);
- do
- ctrl = readl(DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL);
- while (ctrl & (1 << S6_DMA_CHNCTRL_ENABLE));
- }
- static inline void s6dmac_set_stride_skip(u32 dmac, int chan,
- int comchunk, /* 0: disable scatter/gather */
- int srcskip, int dstskip)
- {
- writel(comchunk, DMA_CHNL(dmac, chan) + S6_DMA_CMONCHUNK);
- writel(srcskip, DMA_CHNL(dmac, chan) + S6_DMA_SRCSKIP);
- writel(dstskip, DMA_CHNL(dmac, chan) + S6_DMA_DSTSKIP);
- }
- static inline void s6dmac_enable_chan(u32 dmac, int chan,
- int prio, /* 0 (highest) .. 3 (lowest) */
- int periphxfer, /* <0: disable p.req.line, 0..1: mode */
- int srcinc, int dstinc, /* 0: dont increment src/dst address */
- int comchunk, /* 0: disable scatter/gather */
- int srcskip, int dstskip,
- int burstsize, /* 4 for I2S, 7 for everything else */
- int bandwidthconserve, /* <0: disable, 0..1: select */
- int lowwmark, /* 0..15 */
- int timestamp, /* 0: disable timestamp */
- int enable) /* 0: disable for now */
- {
- writel(1, DMA_CHNL(dmac, chan) + S6_DMA_TERMCNTNB);
- writel(0, DMA_CHNL(dmac, chan) + S6_DMA_TERMCNTTMO);
- writel(lowwmark << S6_DMA_CHNCTRL_LOWWMARK,
- DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL);
- s6dmac_set_stride_skip(dmac, chan, comchunk, srcskip, dstskip);
- writel(((enable ? 1 : 0) << S6_DMA_CHNCTRL_ENABLE) |
- (prio << S6_DMA_CHNCTRL_PRIO) |
- (((periphxfer > 0) ? 1 : 0) << S6_DMA_CHNCTRL_PERIPHXFER) |
- (((periphxfer < 0) ? 0 : 1) << S6_DMA_CHNCTRL_PERIPHENA) |
- ((srcinc ? 1 : 0) << S6_DMA_CHNCTRL_SRCINC) |
- ((dstinc ? 1 : 0) << S6_DMA_CHNCTRL_DSTINC) |
- (burstsize << S6_DMA_CHNCTRL_BURSTLOG) |
- (((bandwidthconserve > 0) ? 1 : 0) << S6_DMA_CHNCTRL_BWCONSEL) |
- (((bandwidthconserve < 0) ? 0 : 1) << S6_DMA_CHNCTRL_BWCONENA) |
- (lowwmark << S6_DMA_CHNCTRL_LOWWMARK) |
- ((timestamp ? 1 : 0) << S6_DMA_CHNCTRL_TSTAMP),
- DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL);
- }
- /* DMA control, per engine */
- static inline unsigned _dmac_addr_index(u32 dmac)
- {
- unsigned i = S6_DMAC_INDEX(dmac);
- if (s6dmac_ctrl[i].dmac != dmac)
- BUG();
- return i;
- }
- static inline void _s6dmac_disable_error_irqs(u32 dmac, u32 mask)
- {
- writel(mask, dmac + S6_DMA_TERMCNTIRQCLR);
- writel(mask, dmac + S6_DMA_PENDCNTIRQCLR);
- writel(mask, dmac + S6_DMA_LOWWMRKIRQCLR);
- writel(readl(dmac + S6_DMA_INTENABLE0)
- & ~((mask << S6_DMA_INT0_UNDER) | (mask << S6_DMA_INT0_OVER)),
- dmac + S6_DMA_INTENABLE0);
- writel(readl(dmac + S6_DMA_INTENABLE1) & ~(mask << S6_DMA_INT1_CHANNEL),
- dmac + S6_DMA_INTENABLE1);
- writel((mask << S6_DMA_INT0_UNDER) | (mask << S6_DMA_INT0_OVER),
- dmac + S6_DMA_INTCLEAR0);
- writel(mask << S6_DMA_INT1_CHANNEL, dmac + S6_DMA_INTCLEAR1);
- }
- /*
- * request channel from specified engine
- * with chan<0, accept any channel
- * further parameters see s6dmac_enable_chan
- * returns < 0 upon error, channel nb otherwise
- */
- static inline int s6dmac_request_chan(u32 dmac, int chan,
- int prio,
- int periphxfer,
- int srcinc, int dstinc,
- int comchunk,
- int srcskip, int dstskip,
- int burstsize,
- int bandwidthconserve,
- int lowwmark,
- int timestamp,
- int enable)
- {
- int r = chan;
- unsigned long flags;
- spinlock_t *spinl = &s6dmac_ctrl[_dmac_addr_index(dmac)].lock;
- spin_lock_irqsave(spinl, flags);
- if (r < 0) {
- r = (readl(dmac + S6_DMA_NEXTFREE) >> S6_DMA_NEXTFREE_CHAN)
- & S6_DMA_NEXTFREE_CHAN_MASK;
- }
- if (r >= s6dmac_ctrl[_dmac_addr_index(dmac)].chan_nb) {
- if (chan < 0)
- r = -EBUSY;
- else
- r = -ENXIO;
- } else if (((readl(dmac + S6_DMA_NEXTFREE) >> S6_DMA_NEXTFREE_ENA)
- >> r) & 1) {
- r = -EBUSY;
- } else {
- s6dmac_enable_chan(dmac, r, prio, periphxfer,
- srcinc, dstinc, comchunk, srcskip, dstskip, burstsize,
- bandwidthconserve, lowwmark, timestamp, enable);
- }
- spin_unlock_irqrestore(spinl, flags);
- return r;
- }
- static inline void s6dmac_put_fifo(u32 dmac, int chan,
- u32 src, u32 dst, u32 size)
- {
- unsigned long flags;
- spinlock_t *spinl = &s6dmac_ctrl[_dmac_addr_index(dmac)].lock;
- spin_lock_irqsave(spinl, flags);
- writel(src, dmac + S6_DMA_DESCRFIFO0);
- writel(dst, dmac + S6_DMA_DESCRFIFO1);
- writel(size, dmac + S6_DMA_DESCRFIFO2);
- writel(chan, dmac + S6_DMA_DESCRFIFO3);
- spin_unlock_irqrestore(spinl, flags);
- }
- static inline u32 s6dmac_channel_enabled(u32 dmac, int chan)
- {
- return readl(DMA_CHNL(dmac, chan) + S6_DMA_CHNCTRL) &
- (1 << S6_DMA_CHNCTRL_ENABLE);
- }
- /*
- * group 1-4 data port channels
- * with port=0..3, nrch=1-4 channels,
- * frrep=0/1 (dis- or enable frame repeat)
- */
- static inline void s6dmac_dp_setup_group(u32 dmac, int port,
- int nrch, int frrep)
- {
- const static u8 mask[4] = {0, 3, 1, 2};
- BUG_ON(dmac != S6_REG_DPDMA);
- if ((port < 0) || (port > 3) || (nrch < 1) || (nrch > 4))
- return;
- writel((mask[nrch - 1] << S6_DMA_DPORTCTRLGRP_NRCHANS)
- | ((frrep ? 1 : 0) << S6_DMA_DPORTCTRLGRP_FRAMEREP),
- dmac + S6_DMA_DPORTCTRLGRP(port));
- }
- static inline void s6dmac_dp_switch_group(u32 dmac, int port, int enable)
- {
- u32 tmp;
- BUG_ON(dmac != S6_REG_DPDMA);
- tmp = readl(dmac + S6_DMA_DPORTCTRLGRP(port));
- if (enable)
- tmp |= (1 << S6_DMA_DPORTCTRLGRP_ENA);
- else
- tmp &= ~(1 << S6_DMA_DPORTCTRLGRP_ENA);
- writel(tmp, dmac + S6_DMA_DPORTCTRLGRP(port));
- }
- extern void s6dmac_put_fifo_cache(u32 dmac, int chan,
- u32 src, u32 dst, u32 size);
- extern void s6dmac_disable_error_irqs(u32 dmac, u32 mask);
- extern u32 s6dmac_int_sources(u32 dmac, u32 channel);
- extern void s6dmac_release_chan(u32 dmac, int chan);
- #endif /* __ASM_XTENSA_S6000_DMAC_H */
|