2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-16 15:20:36 -07:00
# include <linux/types.h>
# include <linux/mm.h>
# include <linux/ioport.h>
# include <linux/init.h>
2009-04-05 13:02:45 +02:00
# include <linux/slab.h>
2005-04-16 15:20:36 -07:00
# include <linux/spinlock.h>
# include <linux/interrupt.h>
2009-04-05 13:02:45 +02:00
# include <linux/platform_device.h>
2011-05-27 09:47:43 -04:00
# include <linux/module.h>
2005-04-16 15:20:36 -07:00
# include <asm/page.h>
# include <asm/amigaints.h>
# include <asm/amigahw.h>
# include "scsi.h"
# include "wd33c93.h"
# include "a3000.h"
2009-03-04 12:06:08 -08:00
2010-04-12 21:55:15 +02:00
struct a3000_hostdata {
struct WD33C93_hostdata wh ;
struct a3000_scsiregs * regs ;
} ;
2009-05-17 13:35:07 +02:00
static irqreturn_t a3000_intr ( int irq , void * data )
2005-04-16 15:20:36 -07:00
{
2009-05-17 13:35:07 +02:00
struct Scsi_Host * instance = data ;
2010-04-12 21:55:15 +02:00
struct a3000_hostdata * hdata = shost_priv ( instance ) ;
unsigned int status = hdata - > regs - > ISTR ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
if ( ! ( status & ISTR_INT_P ) )
return IRQ_NONE ;
2010-04-04 11:00:35 +02:00
if ( status & ISTR_INTS ) {
2009-05-17 13:35:07 +02:00
spin_lock_irqsave ( instance - > host_lock , flags ) ;
wd33c93_intr ( instance ) ;
spin_unlock_irqrestore ( instance - > host_lock , flags ) ;
2005-04-16 15:20:36 -07:00
return IRQ_HANDLED ;
}
2019-10-18 11:18:38 +08:00
pr_warn ( " Non-serviced A3000 SCSI-interrupt? ISTR = %02x \n " , status ) ;
2005-04-16 15:20:36 -07:00
return IRQ_NONE ;
}
2006-09-12 23:49:33 +02:00
static int dma_setup ( struct scsi_cmnd * cmd , int dir_in )
2005-04-16 15:20:36 -07:00
{
2009-05-17 13:35:07 +02:00
struct Scsi_Host * instance = cmd - > device - > host ;
2010-04-12 21:55:15 +02:00
struct a3000_hostdata * hdata = shost_priv ( instance ) ;
struct WD33C93_hostdata * wh = & hdata - > wh ;
struct a3000_scsiregs * regs = hdata - > regs ;
2010-04-04 11:00:35 +02:00
unsigned short cntr = CNTR_PDMD | CNTR_INTEN ;
unsigned long addr = virt_to_bus ( cmd - > SCp . ptr ) ;
/*
* if the physical address has the wrong alignment , or if
* physical address is bad , or if it is a write and at the
* end of a physical memory chunk , then allocate a bounce
* buffer
*/
if ( addr & A3000_XFER_MASK ) {
2010-04-12 21:55:15 +02:00
wh - > dma_bounce_len = ( cmd - > SCp . this_residual + 511 ) & ~ 0x1ff ;
wh - > dma_bounce_buffer = kmalloc ( wh - > dma_bounce_len ,
GFP_KERNEL ) ;
2010-04-04 11:00:35 +02:00
/* can't allocate memory; use PIO */
2010-04-12 21:55:15 +02:00
if ( ! wh - > dma_bounce_buffer ) {
wh - > dma_bounce_len = 0 ;
2010-04-04 11:00:35 +02:00
return 1 ;
}
if ( ! dir_in ) {
/* copy to bounce buffer for a write */
2010-04-12 21:55:15 +02:00
memcpy ( wh - > dma_bounce_buffer , cmd - > SCp . ptr ,
2010-04-04 11:00:39 +02:00
cmd - > SCp . this_residual ) ;
2010-04-04 11:00:35 +02:00
}
2010-04-12 21:55:15 +02:00
addr = virt_to_bus ( wh - > dma_bounce_buffer ) ;
2005-04-16 15:20:36 -07:00
}
2010-04-04 11:00:35 +02:00
/* setup dma direction */
if ( ! dir_in )
cntr | = CNTR_DDIR ;
2005-04-16 15:20:36 -07:00
2010-04-04 11:00:35 +02:00
/* remember direction */
2010-04-12 21:55:15 +02:00
wh - > dma_dir = dir_in ;
2005-04-16 15:20:36 -07:00
2009-05-17 12:17:23 +02:00
regs - > CNTR = cntr ;
2005-04-16 15:20:36 -07:00
2010-04-04 11:00:35 +02:00
/* setup DMA *physical* address */
2009-05-17 12:17:23 +02:00
regs - > ACR = addr ;
2005-04-16 15:20:36 -07:00
2010-04-04 11:00:35 +02:00
if ( dir_in ) {
/* invalidate any cache */
cache_clear ( addr , cmd - > SCp . this_residual ) ;
} else {
/* push any dirty cache */
cache_push ( addr , cmd - > SCp . this_residual ) ;
}
2005-04-16 15:20:36 -07:00
2010-04-04 11:00:35 +02:00
/* start DMA */
mb ( ) ; /* make sure setup is completed */
2009-05-17 12:17:23 +02:00
regs - > ST_DMA = 1 ;
2010-04-04 11:00:35 +02:00
mb ( ) ; /* make sure DMA has started before next IO */
2005-04-16 15:20:36 -07:00
2010-04-04 11:00:35 +02:00
/* return success */
return 0 ;
2005-04-16 15:20:36 -07:00
}
2006-09-12 23:49:33 +02:00
static void dma_stop ( struct Scsi_Host * instance , struct scsi_cmnd * SCpnt ,
int status )
2005-04-16 15:20:36 -07:00
{
2010-04-12 21:55:15 +02:00
struct a3000_hostdata * hdata = shost_priv ( instance ) ;
struct WD33C93_hostdata * wh = & hdata - > wh ;
struct a3000_scsiregs * regs = hdata - > regs ;
2010-04-04 11:00:39 +02:00
2010-04-04 11:00:35 +02:00
/* disable SCSI interrupts */
unsigned short cntr = CNTR_PDMD ;
2010-04-12 21:55:15 +02:00
if ( ! wh - > dma_dir )
2010-04-04 11:00:35 +02:00
cntr | = CNTR_DDIR ;
2009-05-17 12:17:23 +02:00
regs - > CNTR = cntr ;
2010-04-04 11:00:35 +02:00
mb ( ) ; /* make sure CNTR is updated before next IO */
/* flush if we were reading */
2010-04-12 21:55:15 +02:00
if ( wh - > dma_dir ) {
2009-05-17 12:17:23 +02:00
regs - > FLUSH = 1 ;
2010-04-04 11:00:35 +02:00
mb ( ) ; /* don't allow prefetch */
2009-05-17 12:17:23 +02:00
while ( ! ( regs - > ISTR & ISTR_FE_FLG ) )
2010-04-04 11:00:35 +02:00
barrier ( ) ;
mb ( ) ; /* no IO until FLUSH is done */
}
/* clear a possible interrupt */
/* I think that this CINT is only necessary if you are
* using the terminal count features . HM 7 Mar 1994
*/
2009-05-17 12:17:23 +02:00
regs - > CINT = 1 ;
2010-04-04 11:00:35 +02:00
/* stop DMA */
2009-05-17 12:17:23 +02:00
regs - > SP_DMA = 1 ;
2010-04-04 11:00:35 +02:00
mb ( ) ; /* make sure DMA is stopped before next IO */
/* restore the CONTROL bits (minus the direction flag) */
2009-05-17 12:17:23 +02:00
regs - > CNTR = CNTR_PDMD | CNTR_INTEN ;
2010-04-04 11:00:35 +02:00
mb ( ) ; /* make sure CNTR is updated before next IO */
/* copy from a bounce buffer, if necessary */
2010-04-12 21:55:15 +02:00
if ( status & & wh - > dma_bounce_buffer ) {
2010-04-04 11:00:35 +02:00
if ( SCpnt ) {
2010-04-12 21:55:15 +02:00
if ( wh - > dma_dir & & SCpnt )
memcpy ( SCpnt - > SCp . ptr , wh - > dma_bounce_buffer ,
2010-04-04 11:00:35 +02:00
SCpnt - > SCp . this_residual ) ;
2010-04-12 21:55:15 +02:00
kfree ( wh - > dma_bounce_buffer ) ;
wh - > dma_bounce_buffer = NULL ;
wh - > dma_bounce_len = 0 ;
2010-04-04 11:00:35 +02:00
} else {
2010-04-12 21:55:15 +02:00
kfree ( wh - > dma_bounce_buffer ) ;
wh - > dma_bounce_buffer = NULL ;
wh - > dma_bounce_len = 0 ;
2010-04-04 11:00:35 +02:00
}
2005-04-16 15:20:36 -07:00
}
}
2009-04-05 13:02:45 +02:00
static struct scsi_host_template amiga_a3000_scsi_template = {
. module = THIS_MODULE ,
. name = " Amiga 3000 built-in SCSI " ,
2013-03-31 00:30:35 -04:00
. show_info = wd33c93_show_info ,
. write_info = wd33c93_write_info ,
2009-04-05 13:02:45 +02:00
. proc_name = " A3000 " ,
. queuecommand = wd33c93_queuecommand ,
. eh_abort_handler = wd33c93_abort ,
. eh_host_reset_handler = wd33c93_host_reset ,
. can_queue = CAN_QUEUE ,
. this_id = 7 ,
. sg_tablesize = SG_ALL ,
. cmd_per_lun = CMD_PER_LUN ,
} ;
static int __init amiga_a3000_scsi_probe ( struct platform_device * pdev )
2005-04-16 15:20:36 -07:00
{
2009-04-05 13:02:45 +02:00
struct resource * res ;
2009-05-17 13:35:07 +02:00
struct Scsi_Host * instance ;
2009-04-05 13:02:45 +02:00
int error ;
2009-05-17 21:05:53 +02:00
struct a3000_scsiregs * regs ;
2009-04-05 13:02:45 +02:00
wd33c93_regs wdregs ;
2010-04-12 21:55:15 +02:00
struct a3000_hostdata * hdata ;
2010-04-04 11:00:35 +02:00
2009-04-05 13:02:45 +02:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - ENODEV ;
2010-04-04 11:00:35 +02:00
2009-04-05 13:02:45 +02:00
if ( ! request_mem_region ( res - > start , resource_size ( res ) , " wd33c93 " ) )
return - EBUSY ;
2010-04-04 11:00:35 +02:00
2009-04-05 13:02:45 +02:00
instance = scsi_host_alloc ( & amiga_a3000_scsi_template ,
2010-04-12 21:55:15 +02:00
sizeof ( struct a3000_hostdata ) ) ;
2009-04-05 13:02:45 +02:00
if ( ! instance ) {
error = - ENOMEM ;
goto fail_alloc ;
}
2010-04-04 11:00:35 +02:00
2009-05-17 13:35:07 +02:00
instance - > irq = IRQ_AMIGA_PORTS ;
2009-04-05 13:02:45 +02:00
2011-01-09 11:03:43 +01:00
regs = ZTWO_VADDR ( res - > start ) ;
2009-05-17 12:17:23 +02:00
regs - > DAWR = DAWR_A3000 ;
2009-04-05 13:02:45 +02:00
2009-05-17 12:17:23 +02:00
wdregs . SASR = & regs - > SASR ;
wdregs . SCMD = & regs - > SCMD ;
2009-04-05 13:02:45 +02:00
2009-05-17 13:35:07 +02:00
hdata = shost_priv ( instance ) ;
2010-04-12 21:55:15 +02:00
hdata - > wh . no_sync = 0xff ;
hdata - > wh . fast = 0 ;
hdata - > wh . dma_mode = CTRL_DMA ;
hdata - > regs = regs ;
2009-04-05 13:02:45 +02:00
2009-05-17 13:35:07 +02:00
wd33c93_init ( instance , wdregs , dma_setup , dma_stop , WD33C93_FS_12_15 ) ;
2009-04-05 13:02:45 +02:00
error = request_irq ( IRQ_AMIGA_PORTS , a3000_intr , IRQF_SHARED ,
" A3000 SCSI " , instance ) ;
if ( error )
2010-04-04 11:00:35 +02:00
goto fail_irq ;
2009-04-05 13:02:45 +02:00
2009-05-17 12:17:23 +02:00
regs - > CNTR = CNTR_PDMD | CNTR_INTEN ;
2010-04-04 11:00:35 +02:00
2009-04-05 13:02:45 +02:00
error = scsi_add_host ( instance , NULL ) ;
if ( error )
goto fail_host ;
2005-04-16 15:20:36 -07:00
2009-04-05 13:02:45 +02:00
platform_set_drvdata ( pdev , instance ) ;
scsi_scan_host ( instance ) ;
2010-04-04 11:00:35 +02:00
return 0 ;
2009-04-05 13:02:45 +02:00
fail_host :
free_irq ( IRQ_AMIGA_PORTS , instance ) ;
fail_irq :
scsi_host_put ( instance ) ;
fail_alloc :
release_mem_region ( res - > start , resource_size ( res ) ) ;
return error ;
2005-04-16 15:20:36 -07:00
}
2009-04-05 13:02:45 +02:00
static int __exit amiga_a3000_scsi_remove ( struct platform_device * pdev )
2005-04-16 15:20:36 -07:00
{
2009-04-05 13:02:45 +02:00
struct Scsi_Host * instance = platform_get_drvdata ( pdev ) ;
2010-04-12 21:55:15 +02:00
struct a3000_hostdata * hdata = shost_priv ( instance ) ;
2009-04-05 13:02:45 +02:00
struct resource * res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2005-05-28 07:56:31 -04:00
2010-04-12 21:55:15 +02:00
hdata - > regs - > CNTR = 0 ;
2009-04-05 13:02:45 +02:00
scsi_remove_host ( instance ) ;
free_irq ( IRQ_AMIGA_PORTS , instance ) ;
scsi_host_put ( instance ) ;
release_mem_region ( res - > start , resource_size ( res ) ) ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
2009-04-05 13:02:45 +02:00
static struct platform_driver amiga_a3000_scsi_driver = {
. remove = __exit_p ( amiga_a3000_scsi_remove ) ,
. driver = {
. name = " amiga-a3000-scsi " ,
} ,
2005-04-16 15:20:36 -07:00
} ;
2013-05-02 15:30:31 +09:00
module_platform_driver_probe ( amiga_a3000_scsi_driver , amiga_a3000_scsi_probe ) ;
2005-04-16 15:20:36 -07:00
2009-04-05 13:02:45 +02:00
MODULE_DESCRIPTION ( " Amiga 3000 built-in SCSI " ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
2009-04-05 13:02:45 +02:00
MODULE_ALIAS ( " platform:amiga-a3000-scsi " ) ;