2005-04-16 15:20:36 -07:00
/*
2008-02-01 23:09:33 +01:00
* Atari Falcon IDE Driver
2005-04-16 15:20:36 -07:00
*
* Created 12 Jul 1997 by Geert Uytterhoeven
*
* 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 .
*/
2007-07-20 04:33:18 +01:00
# include <linux/module.h>
2005-04-16 15:20:36 -07:00
# include <linux/types.h>
# include <linux/mm.h>
# include <linux/interrupt.h>
# include <linux/blkdev.h>
# include <linux/ide.h>
# include <linux/init.h>
2019-11-06 15:47:29 +13:00
# include <linux/platform_device.h>
2005-04-16 15:20:36 -07:00
# include <asm/setup.h>
# include <asm/atarihw.h>
# include <asm/atariints.h>
# include <asm/atari_stdma.h>
2009-04-08 14:12:48 +02:00
# include <asm/ide.h>
2005-04-16 15:20:36 -07:00
2008-04-26 17:36:38 +02:00
# define DRV_NAME "falconide"
2005-04-16 15:20:36 -07:00
/*
2019-11-06 15:47:29 +13:00
* Offsets from base address
2005-04-16 15:20:36 -07:00
*/
# define ATA_HD_CONTROL 0x39
/*
* falconide_intr_lock is used to obtain access to the IDE interrupt ,
* which is shared between several drivers .
*/
2009-03-27 12:46:22 +01:00
static int falconide_intr_lock ;
static void falconide_release_lock ( void )
{
if ( falconide_intr_lock = = 0 ) {
printk ( KERN_ERR " %s: bug \n " , __func__ ) ;
return ;
}
falconide_intr_lock = 0 ;
stdma_release ( ) ;
}
static void falconide_get_lock ( irq_handler_t handler , void * data )
{
if ( falconide_intr_lock = = 0 ) {
if ( in_interrupt ( ) > 0 )
panic ( " Falcon IDE hasn't ST-DMA lock in interrupt " ) ;
stdma_lock ( handler , data ) ;
falconide_intr_lock = 1 ;
}
}
2005-04-16 15:20:36 -07:00
2009-03-27 12:46:38 +01:00
static void falconide_input_data ( ide_drive_t * drive , struct ide_cmd * cmd ,
2008-04-28 23:44:36 +02:00
void * buf , unsigned int len )
2008-04-28 23:44:36 +02:00
{
2008-04-28 23:44:36 +02:00
unsigned long data_addr = drive - > hwif - > io_ports . data_addr ;
2008-04-28 23:44:36 +02:00
2009-04-08 14:12:48 +02:00
if ( drive - > media = = ide_disk & & cmd & & ( cmd - > tf_flags & IDE_TFLAG_FS ) ) {
__ide_mm_insw ( data_addr , buf , ( len + 1 ) / 2 ) ;
return ;
}
2008-04-28 23:44:36 +02:00
2009-03-27 12:46:23 +01:00
raw_insw_swapw ( ( u16 * ) data_addr , buf , ( len + 1 ) / 2 ) ;
2008-04-28 23:44:36 +02:00
}
2009-03-27 12:46:38 +01:00
static void falconide_output_data ( ide_drive_t * drive , struct ide_cmd * cmd ,
2008-04-28 23:44:36 +02:00
void * buf , unsigned int len )
2008-04-28 23:44:36 +02:00
{
2008-04-28 23:44:36 +02:00
unsigned long data_addr = drive - > hwif - > io_ports . data_addr ;
2009-04-08 14:12:48 +02:00
if ( drive - > media = = ide_disk & & cmd & & ( cmd - > tf_flags & IDE_TFLAG_FS ) ) {
__ide_mm_outsw ( data_addr , buf , ( len + 1 ) / 2 ) ;
return ;
}
2008-04-28 23:44:36 +02:00
2009-03-27 12:46:23 +01:00
raw_outsw_swapw ( ( u16 * ) data_addr , buf , ( len + 1 ) / 2 ) ;
2008-04-28 23:44:36 +02:00
}
2008-07-23 19:55:56 +02:00
/* Atari has a byte-swapped IDE interface */
static const struct ide_tp_ops falconide_tp_ops = {
. exec_command = ide_exec_command ,
. read_status = ide_read_status ,
. read_altstatus = ide_read_altstatus ,
2009-03-31 20:15:30 +02:00
. write_devctl = ide_write_devctl ,
2008-07-23 19:55:56 +02:00
2009-03-31 20:15:32 +02:00
. dev_select = ide_dev_select ,
2008-07-23 19:55:56 +02:00
. tf_load = ide_tf_load ,
. tf_read = ide_tf_read ,
. input_data = falconide_input_data ,
. output_data = falconide_output_data ,
} ;
static const struct ide_port_info falconide_port_info = {
2009-03-27 12:46:22 +01:00
. get_lock = falconide_get_lock ,
. release_lock = falconide_release_lock ,
2008-07-23 19:55:56 +02:00
. tp_ops = & falconide_tp_ops ,
2009-03-27 12:46:23 +01:00
. host_flags = IDE_HFLAG_MMIO | IDE_HFLAG_SERIALIZE |
IDE_HFLAG_NO_DMA ,
2009-03-27 12:46:27 +01:00
. irq_flags = IRQF_SHARED ,
2009-05-17 19:12:22 +02:00
. chipset = ide_generic ,
2008-07-23 19:55:56 +02:00
} ;
2019-11-06 15:47:29 +13:00
static void __init falconide_setup_ports ( struct ide_hw * hw , unsigned long base )
2008-02-06 02:57:50 +01:00
{
int i ;
memset ( hw , 0 , sizeof ( * hw ) ) ;
2019-11-06 15:47:29 +13:00
hw - > io_ports . data_addr = base ;
2008-02-06 02:57:50 +01:00
for ( i = 1 ; i < 8 ; i + + )
2019-11-06 15:47:29 +13:00
hw - > io_ports_array [ i ] = base + 1 + i * 4 ;
2008-02-06 02:57:50 +01:00
2019-11-06 15:47:29 +13:00
hw - > io_ports . ctl_addr = base + ATA_HD_CONTROL ;
2008-02-06 02:57:50 +01:00
hw - > irq = IRQ_MFP_IDE ;
}
2005-04-16 15:20:36 -07:00
/*
* Probe for a Falcon IDE interface
*/
2019-11-06 15:47:29 +13:00
static int __init falconide_init ( struct platform_device * pdev )
2005-04-16 15:20:36 -07:00
{
2019-11-06 15:47:29 +13:00
struct resource * res ;
2008-07-23 19:55:57 +02:00
struct ide_host * host ;
2009-05-17 19:12:25 +02:00
struct ide_hw hw , * hws [ ] = { & hw } ;
2019-11-06 15:47:29 +13:00
unsigned long base ;
2008-07-23 19:55:59 +02:00
int rc ;
2005-04-16 15:20:36 -07:00
2019-11-06 15:47:29 +13:00
dev_info ( & pdev - > dev , " Atari Falcon IDE controller \n " ) ;
2008-02-06 02:57:50 +01:00
2019-11-06 15:47:29 +13:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res )
return - ENODEV ;
2008-01-26 20:13:09 +01:00
2019-11-06 15:47:29 +13:00
if ( ! devm_request_mem_region ( & pdev - > dev , res - > start ,
resource_size ( res ) , DRV_NAME ) ) {
dev_err ( & pdev - > dev , " resources busy \n " ) ;
2008-04-26 17:36:38 +02:00
return - EBUSY ;
}
2019-11-06 15:47:29 +13:00
base = ( unsigned long ) res - > start ;
falconide_setup_ports ( & hw , base ) ;
2005-04-16 15:20:36 -07:00
2009-05-17 19:12:24 +02:00
host = ide_host_alloc ( & falconide_port_info , hws , 1 ) ;
2008-07-23 19:55:59 +02:00
if ( host = = NULL ) {
rc = - ENOMEM ;
goto err ;
2008-01-26 20:13:06 +01:00
}
2008-01-26 20:13:07 +01:00
2009-03-27 12:46:22 +01:00
falconide_get_lock ( NULL , NULL ) ;
2008-07-23 19:55:59 +02:00
rc = ide_host_register ( host , & falconide_port_info , hws ) ;
2009-03-27 12:46:22 +01:00
falconide_release_lock ( ) ;
2008-07-23 19:55:59 +02:00
if ( rc )
goto err_free ;
2008-02-06 02:57:50 +01:00
return 0 ;
2008-07-23 19:55:59 +02:00
err_free :
ide_host_free ( host ) ;
err :
2019-11-06 15:47:29 +13:00
release_mem_region ( res - > start , resource_size ( res ) ) ;
2008-07-23 19:55:59 +02:00
return rc ;
2005-04-16 15:20:36 -07:00
}
2008-01-26 20:13:07 +01:00
2019-11-06 15:47:29 +13:00
static int falconide_remove ( struct platform_device * pdev )
{
struct ide_host * host = dev_get_drvdata ( & pdev - > dev ) ;
ide_host_remove ( host ) ;
return 0 ;
}
static struct platform_driver ide_falcon_driver = {
. remove = falconide_remove ,
. driver = {
. name = " atari-falcon-ide " ,
} ,
} ;
module_platform_driver_probe ( ide_falcon_driver , falconide_init ) ;
2008-04-02 21:22:04 +02:00
2019-11-06 15:47:29 +13:00
MODULE_AUTHOR ( " Geert Uytterhoeven " ) ;
MODULE_DESCRIPTION ( " low-level driver for Atari Falcon IDE " ) ;
2008-04-02 21:22:04 +02:00
MODULE_LICENSE ( " GPL " ) ;
2019-11-06 15:47:29 +13:00
MODULE_ALIAS ( " platform:atari-falcon-ide " ) ;