2008-12-02 21:33:47 +02:00
/*
* MUSB OTG driver - support for Mentor ' s DMA controller
*
* Copyright 2005 Mentor Graphics Corporation
* Copyright ( C ) 2005 - 2007 by Texas Instruments
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* 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 .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA
* 02110 - 1301 USA
*
* THIS SOFTWARE IS PROVIDED " AS IS " AND ANY EXPRESS OR IMPLIED
* WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED . IN
* NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT , INDIRECT ,
* INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF
* USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*
*/
2011-01-27 16:39:40 -08:00
# if defined(CONFIG_SOC_OMAP2430) || defined(CONFIG_SOC_OMAP3430)
2008-12-02 21:33:47 +02:00
# include "omap2430.h"
# endif
# ifndef CONFIG_BLACKFIN
# define MUSB_HSDMA_BASE 0x200
# define MUSB_HSDMA_INTR (MUSB_HSDMA_BASE + 0)
# define MUSB_HSDMA_CONTROL 0x4
# define MUSB_HSDMA_ADDRESS 0x8
# define MUSB_HSDMA_COUNT 0xc
# define MUSB_HSDMA_CHANNEL_OFFSET(_bchannel, _offset) \
( MUSB_HSDMA_BASE + ( _bchannel < < 4 ) + _offset )
# define musb_read_hsdma_addr(mbase, bchannel) \
musb_readl ( mbase , \
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDRESS ) )
# define musb_write_hsdma_addr(mbase, bchannel, addr) \
musb_writel ( mbase , \
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDRESS ) , \
addr )
2009-12-28 13:40:35 +02:00
# define musb_read_hsdma_count(mbase, bchannel) \
musb_readl ( mbase , \
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT ) )
2008-12-02 21:33:47 +02:00
# define musb_write_hsdma_count(mbase, bchannel, len) \
musb_writel ( mbase , \
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT ) , \
len )
# else
# define MUSB_HSDMA_BASE 0x400
# define MUSB_HSDMA_INTR (MUSB_HSDMA_BASE + 0)
# define MUSB_HSDMA_CONTROL 0x04
# define MUSB_HSDMA_ADDR_LOW 0x08
# define MUSB_HSDMA_ADDR_HIGH 0x0C
# define MUSB_HSDMA_COUNT_LOW 0x10
# define MUSB_HSDMA_COUNT_HIGH 0x14
# define MUSB_HSDMA_CHANNEL_OFFSET(_bchannel, _offset) \
( MUSB_HSDMA_BASE + ( _bchannel * 0x20 ) + _offset )
static inline u32 musb_read_hsdma_addr ( void __iomem * mbase , u8 bchannel )
{
u32 addr = musb_readw ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDR_HIGH ) ) ;
addr = addr < < 16 ;
addr | = musb_readw ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDR_LOW ) ) ;
return addr ;
}
static inline void musb_write_hsdma_addr ( void __iomem * mbase ,
u8 bchannel , dma_addr_t dma_addr )
{
musb_writew ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDR_LOW ) ,
2011-01-05 17:36:41 +08:00
dma_addr ) ;
2008-12-02 21:33:47 +02:00
musb_writew ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_ADDR_HIGH ) ,
2011-01-05 17:36:41 +08:00
( dma_addr > > 16 ) ) ;
2008-12-02 21:33:47 +02:00
}
2009-12-28 13:40:35 +02:00
static inline u32 musb_read_hsdma_count ( void __iomem * mbase , u8 bchannel )
{
2011-01-05 17:36:41 +08:00
u32 count = musb_readw ( mbase ,
2009-12-28 13:40:35 +02:00
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT_HIGH ) ) ;
2011-01-05 17:36:41 +08:00
count = count < < 16 ;
count | = musb_readw ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT_LOW ) ) ;
return count ;
2009-12-28 13:40:35 +02:00
}
2008-12-02 21:33:47 +02:00
static inline void musb_write_hsdma_count ( void __iomem * mbase ,
u8 bchannel , u32 len )
{
2011-01-05 17:36:41 +08:00
musb_writew ( mbase ,
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT_LOW ) , len ) ;
musb_writew ( mbase ,
2008-12-02 21:33:47 +02:00
MUSB_HSDMA_CHANNEL_OFFSET ( bchannel , MUSB_HSDMA_COUNT_HIGH ) ,
2011-01-05 17:36:41 +08:00
( len > > 16 ) ) ;
2008-12-02 21:33:47 +02:00
}
# endif /* CONFIG_BLACKFIN */
/* control register (16-bit): */
# define MUSB_HSDMA_ENABLE_SHIFT 0
# define MUSB_HSDMA_TRANSMIT_SHIFT 1
# define MUSB_HSDMA_MODE1_SHIFT 2
# define MUSB_HSDMA_IRQENABLE_SHIFT 3
# define MUSB_HSDMA_ENDPOINT_SHIFT 4
# define MUSB_HSDMA_BUSERROR_SHIFT 8
# define MUSB_HSDMA_BURSTMODE_SHIFT 9
# define MUSB_HSDMA_BURSTMODE (3 << MUSB_HSDMA_BURSTMODE_SHIFT)
# define MUSB_HSDMA_BURSTMODE_UNSPEC 0
# define MUSB_HSDMA_BURSTMODE_INCR4 1
# define MUSB_HSDMA_BURSTMODE_INCR8 2
# define MUSB_HSDMA_BURSTMODE_INCR16 3
# define MUSB_HSDMA_CHANNELS 8
struct musb_dma_controller ;
struct musb_dma_channel {
struct dma_channel channel ;
struct musb_dma_controller * controller ;
u32 start_addr ;
u32 len ;
u16 max_packet_sz ;
u8 idx ;
u8 epnum ;
u8 transmit ;
} ;
struct musb_dma_controller {
struct dma_controller controller ;
struct musb_dma_channel channel [ MUSB_HSDMA_CHANNELS ] ;
void * private_data ;
void __iomem * base ;
u8 channel_count ;
u8 used_channels ;
u8 irq ;
} ;