2005-04-16 15:20:36 -07:00
/*
* Zorro Bus Services
*
* Copyright ( C ) 1995 - 2003 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 .
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/zorro.h>
# include <linux/bitops.h>
2005-10-30 15:03:48 -08:00
# include <linux/string.h>
2009-04-05 12:40:41 +02:00
# include <linux/platform_device.h>
# include <linux/slab.h>
2005-10-30 15:03:48 -08:00
2005-04-16 15:20:36 -07:00
# include <asm/setup.h>
# include <asm/amigahw.h>
# include "zorro.h"
/*
* Zorro Expansion Devices
*/
2009-04-05 12:40:41 +02:00
unsigned int zorro_num_autocon ;
2005-04-16 15:20:36 -07:00
struct zorro_dev zorro_autocon [ ZORRO_NUM_AUTO ] ;
/*
2009-04-05 12:40:41 +02:00
* Zorro bus
2005-04-16 15:20:36 -07:00
*/
2009-04-05 12:40:41 +02:00
struct zorro_bus {
struct list_head devices ; /* list of devices on this bus */
struct device dev ;
2005-04-16 15:20:36 -07:00
} ;
/*
* Find Zorro Devices
*/
struct zorro_dev * zorro_find_device ( zorro_id id , struct zorro_dev * from )
{
2009-04-05 12:40:41 +02:00
struct zorro_dev * z ;
2005-04-16 15:20:36 -07:00
2009-04-05 12:40:41 +02:00
if ( ! zorro_num_autocon )
return NULL ;
2005-04-16 15:20:36 -07:00
2009-04-05 12:40:41 +02:00
for ( z = from ? from + 1 : & zorro_autocon [ 0 ] ;
z < zorro_autocon + zorro_num_autocon ;
z + + )
if ( id = = ZORRO_WILDCARD | | id = = z - > id )
return z ;
return NULL ;
2005-04-16 15:20:36 -07:00
}
2009-04-05 12:40:41 +02:00
EXPORT_SYMBOL ( zorro_find_device ) ;
2005-04-16 15:20:36 -07:00
/*
* Bitmask indicating portions of available Zorro II RAM that are unused
* by the system . Every bit represents a 64 K chunk , for a maximum of 8 MB
* ( 128 chunks , physical 0x00200000 - 0x009fffff ) .
*
* If you want to use ( = allocate ) portions of this RAM , you should clear
* the corresponding bits .
*
* Possible uses :
* - z2ram device
* - SCSI DMA bounce buffers
*
* FIXME : use the normal resource management
*/
DECLARE_BITMAP ( zorro_unused_z2ram , 128 ) ;
2009-04-05 12:40:41 +02:00
EXPORT_SYMBOL ( zorro_unused_z2ram ) ;
2005-04-16 15:20:36 -07:00
static void __init mark_region ( unsigned long start , unsigned long end ,
int flag )
{
if ( flag )
2009-04-05 12:40:41 +02:00
start + = Z2RAM_CHUNKMASK ;
2005-04-16 15:20:36 -07:00
else
2009-04-05 12:40:41 +02:00
end + = Z2RAM_CHUNKMASK ;
start & = ~ Z2RAM_CHUNKMASK ;
end & = ~ Z2RAM_CHUNKMASK ;
if ( end < = Z2RAM_START | | start > = Z2RAM_END )
return ;
start = start < Z2RAM_START ? 0x00000000 : start - Z2RAM_START ;
end = end > Z2RAM_END ? Z2RAM_SIZE : end - Z2RAM_START ;
while ( start < end ) {
u32 chunk = start > > Z2RAM_CHUNKSHIFT ;
if ( flag )
set_bit ( chunk , zorro_unused_z2ram ) ;
else
clear_bit ( chunk , zorro_unused_z2ram ) ;
start + = Z2RAM_CHUNKSIZE ;
}
2005-04-16 15:20:36 -07:00
}
2009-04-05 12:40:41 +02:00
static struct resource __init * zorro_find_parent_resource (
struct platform_device * bridge , struct zorro_dev * z )
2005-04-16 15:20:36 -07:00
{
2009-04-05 12:40:41 +02:00
int i ;
2005-04-16 15:20:36 -07:00
2009-04-05 12:40:41 +02:00
for ( i = 0 ; i < bridge - > num_resources ; i + + ) {
struct resource * r = & bridge - > resource [ i ] ;
if ( zorro_resource_start ( z ) > = r - > start & &
zorro_resource_end ( z ) < = r - > end )
return r ;
}
return & iomem_resource ;
2005-04-16 15:20:36 -07:00
}
2009-04-05 12:40:41 +02:00
static int __init amiga_zorro_probe ( struct platform_device * pdev )
2005-04-16 15:20:36 -07:00
{
2009-04-05 12:40:41 +02:00
struct zorro_bus * bus ;
struct zorro_dev * z ;
struct resource * r ;
unsigned int i ;
int error ;
/* Initialize the Zorro bus */
bus = kzalloc ( sizeof ( * bus ) , GFP_KERNEL ) ;
if ( ! bus )
return - ENOMEM ;
INIT_LIST_HEAD ( & bus - > devices ) ;
bus - > dev . parent = & pdev - > dev ;
dev_set_name ( & bus - > dev , " zorro " ) ;
error = device_register ( & bus - > dev ) ;
2008-12-30 14:21:19 +01:00
if ( error ) {
2009-04-05 12:40:41 +02:00
pr_err ( " Zorro: Error registering zorro_bus \n " ) ;
2010-09-19 16:55:21 +04:00
put_device ( & bus - > dev ) ;
2009-04-05 12:40:41 +02:00
kfree ( bus ) ;
return error ;
2008-12-30 14:21:19 +01:00
}
2009-04-05 12:40:41 +02:00
platform_set_drvdata ( pdev , bus ) ;
pr_info ( " Zorro: Probing AutoConfig expansion devices: %u device%s \n " ,
zorro_num_autocon , zorro_num_autocon = = 1 ? " " : " s " ) ;
2011-09-22 21:47:38 +02:00
/* First identify all devices ... */
2009-04-05 12:40:41 +02:00
for ( i = 0 ; i < zorro_num_autocon ; i + + ) {
z = & zorro_autocon [ i ] ;
z - > id = ( z - > rom . er_Manufacturer < < 16 ) | ( z - > rom . er_Product < < 8 ) ;
if ( z - > id = = ZORRO_PROD_GVP_EPC_BASE ) {
/* GVP quirk */
unsigned long magic = zorro_resource_start ( z ) + 0x8000 ;
z - > id | = * ( u16 * ) ZTWO_VADDR ( magic ) & GVP_PRODMASK ;
}
sprintf ( z - > name , " Zorro device %08x " , z - > id ) ;
zorro_name_device ( z ) ;
z - > resource . name = z - > name ;
r = zorro_find_parent_resource ( pdev , z ) ;
error = request_resource ( r , & z - > resource ) ;
if ( error )
dev_err ( & bus - > dev ,
" Address space collision on device %s %pR \n " ,
z - > name , & z - > resource ) ;
dev_set_name ( & z - > dev , " %02x " , i ) ;
z - > dev . parent = & bus - > dev ;
z - > dev . bus = & zorro_bus_type ;
2011-09-22 21:47:38 +02:00
}
/* ... then register them */
for ( i = 0 ; i < zorro_num_autocon ; i + + ) {
z = & zorro_autocon [ i ] ;
2009-04-05 12:40:41 +02:00
error = device_register ( & z - > dev ) ;
if ( error ) {
dev_err ( & bus - > dev , " Error registering device %s \n " ,
z - > name ) ;
2010-09-19 16:55:21 +04:00
put_device ( & z - > dev ) ;
2009-04-05 12:40:41 +02:00
continue ;
}
error = zorro_create_sysfs_dev_files ( z ) ;
if ( error )
dev_err ( & z - > dev , " Error creating sysfs files \n " ) ;
}
/* Mark all available Zorro II memory */
zorro_for_each_dev ( z ) {
if ( z - > rom . er_Type & ERTF_MEMLIST )
mark_region ( zorro_resource_start ( z ) ,
zorro_resource_end ( z ) + 1 , 1 ) ;
}
/* Unmark all used Zorro II memory */
for ( i = 0 ; i < m68k_num_memory ; i + + )
if ( m68k_memory [ i ] . addr < 16 * 1024 * 1024 )
mark_region ( m68k_memory [ i ] . addr ,
m68k_memory [ i ] . addr + m68k_memory [ i ] . size ,
0 ) ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
2009-04-05 12:40:41 +02:00
static struct platform_driver amiga_zorro_driver = {
. driver = {
. name = " amiga-zorro " ,
. owner = THIS_MODULE ,
} ,
} ;
2005-04-16 15:20:36 -07:00
2009-04-05 12:40:41 +02:00
static int __init amiga_zorro_init ( void )
{
return platform_driver_probe ( & amiga_zorro_driver , amiga_zorro_probe ) ;
}
module_init ( amiga_zorro_init ) ;
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;