2009-12-03 03:57:10 +03:00
/*
Mantis PCI bridge driver
2009-12-15 15:13:49 +03:00
Copyright ( C ) Manu Abraham ( abraham . manu @ gmail . com )
2009-12-03 03:57:10 +03:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
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 . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
2009-12-04 11:41:11 +03:00
# include <linux/kernel.h>
2009-12-03 03:57:10 +03:00
# include <asm/page.h>
# include <linux/vmalloc.h>
2009-12-04 11:41:11 +03:00
# include <linux/pci.h>
# include <asm/irq.h>
# include <linux/signal.h>
# include <linux/sched.h>
# include <linux/interrupt.h>
2017-12-28 21:03:51 +03:00
# include <media/dmxdev.h>
# include <media/dvbdev.h>
# include <media/dvb_demux.h>
# include <media/dvb_frontend.h>
# include <media/dvb_net.h>
2009-12-04 11:41:11 +03:00
2009-12-03 03:57:10 +03:00
# include "mantis_common.h"
2009-12-04 11:41:11 +03:00
# include "mantis_reg.h"
# include "mantis_dma.h"
2009-12-03 03:57:10 +03:00
# define RISC_WRITE (0x01 << 28)
# define RISC_JUMP (0x07 << 28)
# define RISC_IRQ (0x01 << 24)
# define RISC_STATUS(status) ((((~status) & 0x0f) << 20) | ((status & 0x0f) << 16))
2010-08-07 15:16:15 +04:00
# define RISC_FLUSH(risc_pos) (risc_pos = 0)
# define RISC_INSTR(risc_pos, opcode) (mantis->risc_cpu[risc_pos++] = cpu_to_le32(opcode))
2009-12-03 03:57:10 +03:00
2009-12-15 14:47:21 +03:00
# define MANTIS_BUF_SIZE (64 * 1024)
2010-08-07 15:16:15 +04:00
# define MANTIS_BLOCK_BYTES (MANTIS_BUF_SIZE / 4)
# define MANTIS_DMA_TR_BYTES (2 * 1024) /* upper limit: 4095 bytes. */
# define MANTIS_BLOCK_COUNT (MANTIS_BUF_SIZE / MANTIS_BLOCK_BYTES)
# define MANTIS_DMA_TR_UNITS (MANTIS_BLOCK_BYTES / MANTIS_DMA_TR_BYTES)
/* MANTIS_BUF_SIZE / MANTIS_DMA_TR_UNITS must not exceed MANTIS_RISC_SIZE (4k RISC cmd buffer) */
# define MANTIS_RISC_SIZE PAGE_SIZE /* RISC program must fit here. */
2009-12-03 03:57:10 +03:00
int mantis_dma_exit ( struct mantis_pci * mantis )
{
if ( mantis - > buf_cpu ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" DMA=0x%lx cpu=0x%p size=%d " ,
( unsigned long ) mantis - > buf_dma ,
mantis - > buf_cpu ,
MANTIS_BUF_SIZE ) ;
pci_free_consistent ( mantis - > pdev , MANTIS_BUF_SIZE ,
mantis - > buf_cpu , mantis - > buf_dma ) ;
mantis - > buf_cpu = NULL ;
}
if ( mantis - > risc_cpu ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" RISC=0x%lx cpu=0x%p size=%lx " ,
( unsigned long ) mantis - > risc_dma ,
mantis - > risc_cpu ,
MANTIS_RISC_SIZE ) ;
pci_free_consistent ( mantis - > pdev , MANTIS_RISC_SIZE ,
mantis - > risc_cpu , mantis - > risc_dma ) ;
mantis - > risc_cpu = NULL ;
}
return 0 ;
}
2009-12-04 11:41:11 +03:00
EXPORT_SYMBOL_GPL ( mantis_dma_exit ) ;
2009-12-03 03:57:10 +03:00
static inline int mantis_alloc_buffers ( struct mantis_pci * mantis )
{
if ( ! mantis - > buf_cpu ) {
mantis - > buf_cpu = pci_alloc_consistent ( mantis - > pdev ,
MANTIS_BUF_SIZE ,
& mantis - > buf_dma ) ;
if ( ! mantis - > buf_cpu ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" DMA buffer allocation failed " ) ;
goto err ;
}
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" DMA=0x%lx cpu=0x%p size=%d " ,
( unsigned long ) mantis - > buf_dma ,
mantis - > buf_cpu , MANTIS_BUF_SIZE ) ;
}
if ( ! mantis - > risc_cpu ) {
mantis - > risc_cpu = pci_alloc_consistent ( mantis - > pdev ,
MANTIS_RISC_SIZE ,
& mantis - > risc_dma ) ;
if ( ! mantis - > risc_cpu ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" RISC program allocation failed " ) ;
mantis_dma_exit ( mantis ) ;
goto err ;
}
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 ,
2009-12-03 03:57:10 +03:00
" RISC=0x%lx cpu=0x%p size=%lx " ,
( unsigned long ) mantis - > risc_dma ,
mantis - > risc_cpu , MANTIS_RISC_SIZE ) ;
}
return 0 ;
err :
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 , " Out of memory (?) ..... " ) ;
2009-12-03 03:57:10 +03:00
return - ENOMEM ;
}
int mantis_dma_init ( struct mantis_pci * mantis )
{
2015-08-10 20:11:41 +03:00
int err ;
2009-12-03 03:57:10 +03:00
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_DEBUG , 1 , " Mantis DMA init " ) ;
2015-08-10 20:11:41 +03:00
err = mantis_alloc_buffers ( mantis ) ;
if ( err < 0 ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_ERROR , 1 , " Error allocating DMA buffer " ) ;
2009-12-03 03:57:10 +03:00
2009-12-04 11:56:35 +03:00
/* Stop RISC Engine */
2009-12-03 03:57:10 +03:00
mmwrite ( 0 , MANTIS_DMA_CTL ) ;
2015-08-11 16:30:24 +03:00
return err ;
2009-12-03 03:57:10 +03:00
}
return 0 ;
}
2009-12-04 11:41:11 +03:00
EXPORT_SYMBOL_GPL ( mantis_dma_init ) ;
2009-12-03 03:57:10 +03:00
static inline void mantis_risc_program ( struct mantis_pci * mantis )
{
u32 buf_pos = 0 ;
2010-08-07 15:16:15 +04:00
u32 line , step ;
u32 risc_pos ;
2009-12-03 03:57:10 +03:00
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_DEBUG , 1 , " Mantis create RISC program " ) ;
2010-08-07 15:16:15 +04:00
RISC_FLUSH ( risc_pos ) ;
dprintk ( MANTIS_DEBUG , 1 , " risc len lines %u, bytes per line %u, bytes per DMA tr %u " ,
MANTIS_BLOCK_COUNT , MANTIS_BLOCK_BYTES , MANTIS_DMA_TR_BYTES ) ;
for ( line = 0 ; line < MANTIS_BLOCK_COUNT ; line + + ) {
for ( step = 0 ; step < MANTIS_DMA_TR_UNITS ; step + + ) {
dprintk ( MANTIS_DEBUG , 1 , " RISC PROG line=[%d], step=[%d] " , line , step ) ;
if ( step = = 0 ) {
RISC_INSTR ( risc_pos , RISC_WRITE |
RISC_IRQ |
RISC_STATUS ( line ) |
MANTIS_DMA_TR_BYTES ) ;
} else {
RISC_INSTR ( risc_pos , RISC_WRITE | MANTIS_DMA_TR_BYTES ) ;
}
RISC_INSTR ( risc_pos , mantis - > buf_dma + buf_pos ) ;
buf_pos + = MANTIS_DMA_TR_BYTES ;
}
2009-12-03 03:57:10 +03:00
}
2010-08-07 15:16:15 +04:00
RISC_INSTR ( risc_pos , RISC_JUMP ) ;
RISC_INSTR ( risc_pos , mantis - > risc_dma ) ;
2009-12-03 03:57:10 +03:00
}
void mantis_dma_start ( struct mantis_pci * mantis )
{
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_DEBUG , 1 , " Mantis Start DMA engine " ) ;
2009-12-03 03:57:10 +03:00
mantis_risc_program ( mantis ) ;
2009-12-03 11:47:11 +03:00
mmwrite ( mantis - > risc_dma , MANTIS_RISC_START ) ;
2009-12-04 11:13:21 +03:00
mmwrite ( mmread ( MANTIS_GPIF_ADDR ) | MANTIS_GPIF_HIFRDWRN , MANTIS_GPIF_ADDR ) ;
2009-12-03 03:57:10 +03:00
mmwrite ( 0 , MANTIS_DMA_CTL ) ;
2010-08-07 15:16:15 +04:00
mantis - > last_block = mantis - > busy_block = 0 ;
2009-12-03 03:57:10 +03:00
2015-06-06 22:58:13 +03:00
mantis_unmask_ints ( mantis , MANTIS_INT_RISCI ) ;
2009-12-03 03:57:10 +03:00
mmwrite ( MANTIS_FIFO_EN | MANTIS_DCAP_EN
| MANTIS_RISC_EN , MANTIS_DMA_CTL ) ;
}
void mantis_dma_stop ( struct mantis_pci * mantis )
{
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_DEBUG , 1 , " Mantis Stop DMA engine " ) ;
2009-12-03 03:57:10 +03:00
2009-12-04 11:13:21 +03:00
mmwrite ( ( mmread ( MANTIS_GPIF_ADDR ) & ( ~ ( MANTIS_GPIF_HIFRDWRN ) ) ) , MANTIS_GPIF_ADDR ) ;
2009-12-03 11:39:22 +03:00
2009-12-03 03:57:10 +03:00
mmwrite ( ( mmread ( MANTIS_DMA_CTL ) & ~ ( MANTIS_FIFO_EN |
MANTIS_DCAP_EN |
MANTIS_RISC_EN ) ) , MANTIS_DMA_CTL ) ;
mmwrite ( mmread ( MANTIS_INT_STAT ) , MANTIS_INT_STAT ) ;
2015-06-06 22:58:13 +03:00
mantis_mask_ints ( mantis , MANTIS_INT_RISCI | MANTIS_INT_RISCEN ) ;
2009-12-03 03:57:10 +03:00
}
void mantis_dma_xfer ( unsigned long data )
{
struct mantis_pci * mantis = ( struct mantis_pci * ) data ;
2009-12-03 11:44:38 +03:00
struct mantis_hwconfig * config = mantis - > hwconfig ;
2009-12-03 03:57:10 +03:00
2010-08-07 15:16:15 +04:00
while ( mantis - > last_block ! = mantis - > busy_block ) {
2009-12-04 11:41:11 +03:00
dprintk ( MANTIS_DEBUG , 1 , " last block=[%d] finished block=[%d] " ,
2010-08-07 15:16:15 +04:00
mantis - > last_block , mantis - > busy_block ) ;
2009-12-03 03:57:10 +03:00
2009-12-15 14:47:21 +03:00
( config - > ts_size ? dvb_dmx_swfilter_204 : dvb_dmx_swfilter )
2009-12-03 03:57:10 +03:00
( & mantis - > demux , & mantis - > buf_cpu [ mantis - > last_block * MANTIS_BLOCK_BYTES ] , MANTIS_BLOCK_BYTES ) ;
mantis - > last_block = ( mantis - > last_block + 1 ) % MANTIS_BLOCK_COUNT ;
}
}