2005-04-16 15:20:36 -07:00
/*
* UniNorth AGPGART routines .
*/
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/init.h>
# include <linux/pagemap.h>
# include <linux/agp_backend.h>
# include <linux/delay.h>
# include <asm/uninorth.h>
# include <asm/pci-bridge.h>
# include <asm/prom.h>
2005-04-16 15:24:19 -07:00
# include <asm/pmac_feature.h>
2005-04-16 15:20:36 -07:00
# include "agp.h"
/*
* NOTES for uninorth3 ( G5 AGP ) supports :
*
* There maybe also possibility to have bigger cache line size for
* agp ( see pmac_pci . c and look for cache line ) . Need to be investigated
* by someone .
*
* PAGE size are hardcoded but this may change , see asm / page . h .
*
* Jerome Glisse < j . glisse @ gmail . com >
*/
static int uninorth_rev ;
static int is_u3 ;
2006-10-04 14:56:44 +02:00
static char __devinitdata * aperture = NULL ;
2005-04-16 15:24:19 -07:00
2005-04-16 15:20:36 -07:00
static int uninorth_fetch_size ( void )
{
2006-10-04 14:56:44 +02:00
int i , size = 0 ;
struct aper_size_info_32 * values =
A_SIZE_32 ( agp_bridge - > driver - > aperture_sizes ) ;
if ( aperture ) {
char * save = aperture ;
size = memparse ( aperture , & aperture ) > > 20 ;
aperture = save ;
for ( i = 0 ; i < agp_bridge - > driver - > num_aperture_sizes ; i + + )
if ( size = = values [ i ] . size )
break ;
if ( i = = agp_bridge - > driver - > num_aperture_sizes ) {
2008-07-30 12:26:51 -07:00
dev_err ( & agp_bridge - > dev - > dev , " invalid aperture size, "
" using default \n " ) ;
2006-10-04 14:56:44 +02:00
size = 0 ;
aperture = NULL ;
2005-04-16 15:20:36 -07:00
}
}
2006-10-04 14:56:44 +02:00
if ( ! size ) {
for ( i = 0 ; i < agp_bridge - > driver - > num_aperture_sizes ; i + + )
if ( values [ i ] . size = = 32 )
break ;
}
2005-04-16 15:20:36 -07:00
2006-10-04 14:56:44 +02:00
agp_bridge - > previous_size =
agp_bridge - > current_size = ( void * ) ( values + i ) ;
agp_bridge - > aperture_size_idx = i ;
return values [ i ] . size ;
2005-04-16 15:20:36 -07:00
}
static void uninorth_tlbflush ( struct agp_memory * mem )
{
u32 ctrl = UNI_N_CFG_GART_ENABLE ;
if ( is_u3 )
ctrl | = U3_N_CFG_GART_PERFRD ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL ,
ctrl | UNI_N_CFG_GART_INVAL ) ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL , ctrl ) ;
if ( uninorth_rev < = 0x30 ) {
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL ,
ctrl | UNI_N_CFG_GART_2xRESET ) ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL ,
ctrl ) ;
}
}
static void uninorth_cleanup ( void )
{
u32 tmp ;
pci_read_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL , & tmp ) ;
if ( ! ( tmp & UNI_N_CFG_GART_ENABLE ) )
return ;
tmp | = UNI_N_CFG_GART_INVAL ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL , tmp ) ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL , 0 ) ;
if ( uninorth_rev < = 0x30 ) {
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL ,
UNI_N_CFG_GART_2xRESET ) ;
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_GART_CTRL ,
0 ) ;
}
}
static int uninorth_configure ( void )
{
struct aper_size_info_32 * current_size ;
2006-02-28 00:54:25 -05:00
2005-04-16 15:20:36 -07:00
current_size = A_SIZE_32 ( agp_bridge - > current_size ) ;
2008-07-30 12:26:51 -07:00
dev_info ( & agp_bridge - > dev - > dev , " configuring for size idx: %d \n " ,
current_size - > size_value ) ;
2006-02-28 00:54:25 -05:00
2005-04-16 15:20:36 -07:00
/* aperture size and gatt addr */
pci_write_config_dword ( agp_bridge - > dev ,
UNI_N_CFG_GART_BASE ,
( agp_bridge - > gatt_bus_addr & 0xfffff000 )
| current_size - > size_value ) ;
/* HACK ALERT
* UniNorth seem to be buggy enough not to handle properly when
* the AGP aperture isn ' t mapped at bus physical address 0
*/
agp_bridge - > gart_bus_addr = 0 ;
# ifdef CONFIG_PPC64
/* Assume U3 or later on PPC64 systems */
/* high 4 bits of GART physical address go in UNI_N_CFG_AGP_BASE */
pci_write_config_dword ( agp_bridge - > dev , UNI_N_CFG_AGP_BASE ,
( agp_bridge - > gatt_bus_addr > > 32 ) & 0xf ) ;
# else
pci_write_config_dword ( agp_bridge - > dev ,
UNI_N_CFG_AGP_BASE , agp_bridge - > gart_bus_addr ) ;
# endif
if ( is_u3 ) {
pci_write_config_dword ( agp_bridge - > dev ,
UNI_N_CFG_GART_DUMMY_PAGE ,
agp_bridge - > scratch_page_real > > 12 ) ;
}
2006-02-28 00:54:25 -05:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
static int uninorth_insert_memory ( struct agp_memory * mem , off_t pg_start ,
int type )
{
int i , j , num_entries ;
void * temp ;
temp = agp_bridge - > current_size ;
num_entries = A_SIZE_32 ( temp ) - > num_entries ;
if ( type ! = 0 | | mem - > type ! = 0 )
/* We know nothing of memory types */
return - EINVAL ;
if ( ( pg_start + mem - > page_count ) > num_entries )
return - EINVAL ;
j = pg_start ;
while ( j < ( pg_start + mem - > page_count ) ) {
if ( agp_bridge - > gatt_table [ j ] )
return - EBUSY ;
j + + ;
}
for ( i = 0 , j = pg_start ; i < mem - > page_count ; i + + , j + + ) {
agp_bridge - > gatt_table [ j ] =
cpu_to_le32 ( ( mem - > memory [ i ] & 0xFFFFF000UL ) | 0x1UL ) ;
flush_dcache_range ( ( unsigned long ) __va ( mem - > memory [ i ] ) ,
( unsigned long ) __va ( mem - > memory [ i ] ) + 0x1000 ) ;
}
( void ) in_le32 ( ( volatile u32 * ) & agp_bridge - > gatt_table [ pg_start ] ) ;
mb ( ) ;
2006-02-28 00:54:25 -05:00
flush_dcache_range ( ( unsigned long ) & agp_bridge - > gatt_table [ pg_start ] ,
2005-04-16 15:20:36 -07:00
( unsigned long ) & agp_bridge - > gatt_table [ pg_start + mem - > page_count ] ) ;
uninorth_tlbflush ( mem ) ;
return 0 ;
}
static int u3_insert_memory ( struct agp_memory * mem , off_t pg_start , int type )
{
int i , num_entries ;
void * temp ;
u32 * gp ;
temp = agp_bridge - > current_size ;
num_entries = A_SIZE_32 ( temp ) - > num_entries ;
if ( type ! = 0 | | mem - > type ! = 0 )
/* We know nothing of memory types */
return - EINVAL ;
if ( ( pg_start + mem - > page_count ) > num_entries )
return - EINVAL ;
gp = ( u32 * ) & agp_bridge - > gatt_table [ pg_start ] ;
for ( i = 0 ; i < mem - > page_count ; + + i ) {
if ( gp [ i ] ) {
2008-07-30 12:26:51 -07:00
dev_info ( & agp_bridge - > dev - > dev ,
" u3_insert_memory: entry 0x%x occupied (%x) \n " ,
i , gp [ i ] ) ;
2005-04-16 15:20:36 -07:00
return - EBUSY ;
}
}
for ( i = 0 ; i < mem - > page_count ; i + + ) {
gp [ i ] = ( mem - > memory [ i ] > > PAGE_SHIFT ) | 0x80000000UL ;
flush_dcache_range ( ( unsigned long ) __va ( mem - > memory [ i ] ) ,
( unsigned long ) __va ( mem - > memory [ i ] ) + 0x1000 ) ;
}
mb ( ) ;
flush_dcache_range ( ( unsigned long ) gp , ( unsigned long ) & gp [ i ] ) ;
uninorth_tlbflush ( mem ) ;
return 0 ;
}
int u3_remove_memory ( struct agp_memory * mem , off_t pg_start , int type )
{
size_t i ;
u32 * gp ;
if ( type ! = 0 | | mem - > type ! = 0 )
/* We know nothing of memory types */
return - EINVAL ;
gp = ( u32 * ) & agp_bridge - > gatt_table [ pg_start ] ;
for ( i = 0 ; i < mem - > page_count ; + + i )
gp [ i ] = 0 ;
mb ( ) ;
flush_dcache_range ( ( unsigned long ) gp , ( unsigned long ) & gp [ i ] ) ;
uninorth_tlbflush ( mem ) ;
return 0 ;
}
static void uninorth_agp_enable ( struct agp_bridge_data * bridge , u32 mode )
{
u32 command , scratch , status ;
int timeout ;
pci_read_config_dword ( bridge - > dev ,
bridge - > capndx + PCI_AGP_STATUS ,
& status ) ;
command = agp_collect_device_status ( bridge , mode , status ) ;
command | = PCI_AGP_COMMAND_AGP ;
2006-02-28 00:54:25 -05:00
2005-04-16 15:20:36 -07:00
if ( uninorth_rev = = 0x21 ) {
/*
* Darwin disable AGP 4 x on this revision , thus we
* may assume it ' s broken . This is an AGP2 controller .
*/
command & = ~ AGPSTAT2_4X ;
}
if ( ( uninorth_rev > = 0x30 ) & & ( uninorth_rev < = 0x33 ) ) {
/*
* We need to to set REQ_DEPTH to 7 for U3 versions 1.0 , 2.1 ,
* 2.2 and 2.3 , Darwin do so .
*/
if ( ( command > > AGPSTAT_RQ_DEPTH_SHIFT ) > 7 )
command = ( command & ~ AGPSTAT_RQ_DEPTH )
| ( 7 < < AGPSTAT_RQ_DEPTH_SHIFT ) ;
}
uninorth_tlbflush ( NULL ) ;
timeout = 0 ;
do {
pci_write_config_dword ( bridge - > dev ,
bridge - > capndx + PCI_AGP_COMMAND ,
command ) ;
pci_read_config_dword ( bridge - > dev ,
bridge - > capndx + PCI_AGP_COMMAND ,
& scratch ) ;
} while ( ( scratch & PCI_AGP_COMMAND_AGP ) = = 0 & & + + timeout < 1000 ) ;
if ( ( scratch & PCI_AGP_COMMAND_AGP ) = = 0 )
2008-07-30 12:26:51 -07:00
dev_err ( & bridge - > dev - > dev , " can't write UniNorth AGP "
" command register \n " ) ;
2005-04-16 15:20:36 -07:00
if ( uninorth_rev > = 0x30 ) {
/* This is an AGP V3 */
2008-03-26 14:10:02 -07:00
agp_device_command ( command , ( status & AGPSTAT_MODE_3_0 ) ! = 0 ) ;
2005-04-16 15:20:36 -07:00
} else {
/* AGP V2 */
2008-03-26 14:10:02 -07:00
agp_device_command ( command , false ) ;
2005-04-16 15:20:36 -07:00
}
uninorth_tlbflush ( NULL ) ;
}
# ifdef CONFIG_PM
2005-04-16 15:24:19 -07:00
/*
* These Power Management routines are _not_ called by the normal PCI PM layer ,
* but directly by the video driver through function pointers in the device
* tree .
*/
static int agp_uninorth_suspend ( struct pci_dev * pdev )
2005-04-16 15:20:36 -07:00
{
2005-04-16 15:24:19 -07:00
struct agp_bridge_data * bridge ;
2005-04-16 15:20:36 -07:00
u32 cmd ;
u8 agp ;
struct pci_dev * device = NULL ;
2005-04-16 15:24:19 -07:00
bridge = agp_find_bridge ( pdev ) ;
if ( bridge = = NULL )
return - ENODEV ;
/* Only one suspend supported */
if ( bridge - > dev_private_data )
2005-04-16 15:20:36 -07:00
return 0 ;
/* turn off AGP on the video chip, if it was enabled */
for_each_pci_dev ( device ) {
/* Don't touch the bridge yet, device first */
if ( device = = pdev )
continue ;
/* Only deal with devices on the same bus here, no Mac has a P2P
* bridge on the AGP port , and mucking around the entire PCI
* tree is source of problems on some machines because of a bug
* in some versions of pci_find_capability ( ) when hitting a dead
* device
*/
if ( device - > bus ! = pdev - > bus )
continue ;
agp = pci_find_capability ( device , PCI_CAP_ID_AGP ) ;
if ( ! agp )
continue ;
pci_read_config_dword ( device , agp + PCI_AGP_COMMAND , & cmd ) ;
if ( ! ( cmd & PCI_AGP_COMMAND_AGP ) )
continue ;
2008-07-30 12:26:51 -07:00
dev_info ( & pdev - > dev , " disabling AGP on device %s \n " ,
pci_name ( device ) ) ;
2005-04-16 15:20:36 -07:00
cmd & = ~ PCI_AGP_COMMAND_AGP ;
pci_write_config_dword ( device , agp + PCI_AGP_COMMAND , cmd ) ;
}
/* turn off AGP on the bridge */
agp = pci_find_capability ( pdev , PCI_CAP_ID_AGP ) ;
pci_read_config_dword ( pdev , agp + PCI_AGP_COMMAND , & cmd ) ;
2006-06-01 20:19:35 -07:00
bridge - > dev_private_data = ( void * ) ( long ) cmd ;
2005-04-16 15:20:36 -07:00
if ( cmd & PCI_AGP_COMMAND_AGP ) {
2008-07-30 12:26:51 -07:00
dev_info ( & pdev - > dev , " disabling AGP on bridge \n " ) ;
2005-04-16 15:20:36 -07:00
cmd & = ~ PCI_AGP_COMMAND_AGP ;
pci_write_config_dword ( pdev , agp + PCI_AGP_COMMAND , cmd ) ;
}
/* turn off the GART */
uninorth_cleanup ( ) ;
return 0 ;
}
static int agp_uninorth_resume ( struct pci_dev * pdev )
{
2005-04-16 15:24:19 -07:00
struct agp_bridge_data * bridge ;
u32 command ;
bridge = agp_find_bridge ( pdev ) ;
if ( bridge = = NULL )
return - ENODEV ;
2006-06-01 20:19:35 -07:00
command = ( long ) bridge - > dev_private_data ;
2005-04-16 15:24:19 -07:00
bridge - > dev_private_data = NULL ;
if ( ! ( command & PCI_AGP_COMMAND_AGP ) )
return 0 ;
uninorth_agp_enable ( bridge , command ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2005-04-16 15:24:19 -07:00
# endif /* CONFIG_PM */
2005-04-16 15:20:36 -07:00
static int uninorth_create_gatt_table ( struct agp_bridge_data * bridge )
{
char * table ;
char * table_end ;
int size ;
int page_order ;
int num_entries ;
int i ;
void * temp ;
struct page * page ;
/* We can't handle 2 level gatt's */
if ( bridge - > driver - > size_type = = LVL2_APER_SIZE )
return - EINVAL ;
table = NULL ;
i = bridge - > aperture_size_idx ;
temp = bridge - > current_size ;
size = page_order = num_entries = 0 ;
do {
size = A_SIZE_32 ( temp ) - > size ;
page_order = A_SIZE_32 ( temp ) - > page_order ;
num_entries = A_SIZE_32 ( temp ) - > num_entries ;
table = ( char * ) __get_free_pages ( GFP_KERNEL , page_order ) ;
if ( table = = NULL ) {
i + + ;
bridge - > current_size = A_IDX32 ( bridge ) ;
} else {
bridge - > aperture_size_idx = i ;
}
} while ( ! table & & ( i < bridge - > driver - > num_aperture_sizes ) ) ;
if ( table = = NULL )
return - ENOMEM ;
table_end = table + ( ( PAGE_SIZE * ( 1 < < page_order ) ) - 1 ) ;
for ( page = virt_to_page ( table ) ; page < = virt_to_page ( table_end ) ; page + + )
SetPageReserved ( page ) ;
bridge - > gatt_table_real = ( u32 * ) table ;
bridge - > gatt_table = ( u32 * ) table ;
2005-03-30 13:17:04 -08:00
bridge - > gatt_bus_addr = virt_to_gart ( table ) ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < num_entries ; i + + )
bridge - > gatt_table [ i ] = 0 ;
flush_dcache_range ( ( unsigned long ) table , ( unsigned long ) table_end ) ;
return 0 ;
}
static int uninorth_free_gatt_table ( struct agp_bridge_data * bridge )
{
int page_order ;
char * table , * table_end ;
void * temp ;
struct page * page ;
temp = bridge - > current_size ;
page_order = A_SIZE_32 ( temp ) - > page_order ;
/* Do not worry about freeing memory, because if this is
* called , then all agp memory is deallocated and removed
* from the table .
*/
table = ( char * ) bridge - > gatt_table_real ;
table_end = table + ( ( PAGE_SIZE * ( 1 < < page_order ) ) - 1 ) ;
for ( page = virt_to_page ( table ) ; page < = virt_to_page ( table_end ) ; page + + )
ClearPageReserved ( page ) ;
free_pages ( ( unsigned long ) bridge - > gatt_table_real , page_order ) ;
return 0 ;
}
void null_cache_flush ( void )
{
mb ( ) ;
}
/* Setup function */
2007-02-22 18:41:28 -05:00
static const struct aper_size_info_32 uninorth_sizes [ 7 ] =
2005-04-16 15:20:36 -07:00
{
#if 0 /* Not sure uninorth supports that high aperture sizes */
{ 256 , 65536 , 6 , 64 } ,
{ 128 , 32768 , 5 , 32 } ,
{ 64 , 16384 , 4 , 16 } ,
2006-02-28 00:54:25 -05:00
# endif
2005-04-16 15:20:36 -07:00
{ 32 , 8192 , 3 , 8 } ,
{ 16 , 4096 , 2 , 4 } ,
{ 8 , 2048 , 1 , 2 } ,
{ 4 , 1024 , 0 , 1 }
} ;
/*
* Not sure that u3 supports that high aperture sizes but it
* would strange if it did not : )
*/
2007-02-22 18:41:28 -05:00
static const struct aper_size_info_32 u3_sizes [ 8 ] =
2005-04-16 15:20:36 -07:00
{
{ 512 , 131072 , 7 , 128 } ,
{ 256 , 65536 , 6 , 64 } ,
{ 128 , 32768 , 5 , 32 } ,
{ 64 , 16384 , 4 , 16 } ,
{ 32 , 8192 , 3 , 8 } ,
{ 16 , 4096 , 2 , 4 } ,
{ 8 , 2048 , 1 , 2 } ,
{ 4 , 1024 , 0 , 1 }
} ;
2007-02-27 14:13:02 +09:00
const struct agp_bridge_driver uninorth_agp_driver = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. aperture_sizes = ( void * ) uninorth_sizes ,
. size_type = U32_APER_SIZE ,
. num_aperture_sizes = 4 ,
. configure = uninorth_configure ,
. fetch_size = uninorth_fetch_size ,
. cleanup = uninorth_cleanup ,
. tlb_flush = uninorth_tlbflush ,
. mask_memory = agp_generic_mask_memory ,
. masks = NULL ,
. cache_flush = null_cache_flush ,
. agp_enable = uninorth_agp_enable ,
. create_gatt_table = uninorth_create_gatt_table ,
. free_gatt_table = uninorth_free_gatt_table ,
. insert_memory = uninorth_insert_memory ,
. remove_memory = agp_generic_remove_memory ,
. alloc_by_type = agp_generic_alloc_by_type ,
. free_by_type = agp_generic_free_by_type ,
. agp_alloc_page = agp_generic_alloc_page ,
2008-08-21 19:15:46 +02:00
. agp_alloc_pages = agp_generic_alloc_pages ,
2005-04-16 15:20:36 -07:00
. agp_destroy_page = agp_generic_destroy_page ,
2008-08-21 19:15:46 +02:00
. agp_destroy_pages = agp_generic_destroy_pages ,
2007-01-23 10:33:43 +01:00
. agp_type_to_mask_type = agp_generic_type_to_mask_type ,
2008-03-26 14:10:02 -07:00
. cant_use_aperture = true ,
2005-04-16 15:20:36 -07:00
} ;
2007-02-27 14:13:02 +09:00
const struct agp_bridge_driver u3_agp_driver = {
2005-04-16 15:20:36 -07:00
. owner = THIS_MODULE ,
. aperture_sizes = ( void * ) u3_sizes ,
. size_type = U32_APER_SIZE ,
. num_aperture_sizes = 8 ,
. configure = uninorth_configure ,
. fetch_size = uninorth_fetch_size ,
. cleanup = uninorth_cleanup ,
. tlb_flush = uninorth_tlbflush ,
. mask_memory = agp_generic_mask_memory ,
. masks = NULL ,
. cache_flush = null_cache_flush ,
. agp_enable = uninorth_agp_enable ,
. create_gatt_table = uninorth_create_gatt_table ,
. free_gatt_table = uninorth_free_gatt_table ,
. insert_memory = u3_insert_memory ,
. remove_memory = u3_remove_memory ,
. alloc_by_type = agp_generic_alloc_by_type ,
. free_by_type = agp_generic_free_by_type ,
. agp_alloc_page = agp_generic_alloc_page ,
2008-08-21 19:15:46 +02:00
. agp_alloc_pages = agp_generic_alloc_pages ,
2005-04-16 15:20:36 -07:00
. agp_destroy_page = agp_generic_destroy_page ,
2008-08-21 19:15:46 +02:00
. agp_destroy_paged = agp_generic_destroy_pages ,
2007-01-23 10:33:43 +01:00
. agp_type_to_mask_type = agp_generic_type_to_mask_type ,
2008-03-26 14:10:02 -07:00
. cant_use_aperture = true ,
. needs_scratch_page = true ,
2005-04-16 15:20:36 -07:00
} ;
static struct agp_device_ids uninorth_agp_device_ids [ ] __devinitdata = {
{
. device_id = PCI_DEVICE_ID_APPLE_UNI_N_AGP ,
. chipset_name = " UniNorth " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_UNI_N_AGP_P ,
. chipset_name = " UniNorth/Pangea " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_UNI_N_AGP15 ,
. chipset_name = " UniNorth 1.5 " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_UNI_N_AGP2 ,
. chipset_name = " UniNorth 2 " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_U3_AGP ,
. chipset_name = " U3 " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_U3L_AGP ,
. chipset_name = " U3L " ,
} ,
{
. device_id = PCI_DEVICE_ID_APPLE_U3H_AGP ,
. chipset_name = " U3H " ,
} ,
2005-11-13 16:06:48 -08:00
{
. device_id = PCI_DEVICE_ID_APPLE_IPID2_AGP ,
. chipset_name = " UniNorth/Intrepid2 " ,
} ,
2005-04-16 15:20:36 -07:00
} ;
static int __devinit agp_uninorth_probe ( struct pci_dev * pdev ,
const struct pci_device_id * ent )
{
struct agp_device_ids * devs = uninorth_agp_device_ids ;
struct agp_bridge_data * bridge ;
struct device_node * uninorth_node ;
u8 cap_ptr ;
int j ;
cap_ptr = pci_find_capability ( pdev , PCI_CAP_ID_AGP ) ;
if ( cap_ptr = = 0 )
return - ENODEV ;
/* probe for known chipsets */
for ( j = 0 ; devs [ j ] . chipset_name ! = NULL ; + + j ) {
if ( pdev - > device = = devs [ j ] . device_id ) {
2008-07-30 12:26:51 -07:00
dev_info ( & pdev - > dev , " Apple %s chipset \n " ,
devs [ j ] . chipset_name ) ;
2005-04-16 15:20:36 -07:00
goto found ;
}
}
2008-07-30 12:26:51 -07:00
dev_err ( & pdev - > dev , " unsupported Apple chipset [%04x/%04x] \n " ,
pdev - > vendor , pdev - > device ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
found :
/* Set revision to 0 if we could not read it. */
uninorth_rev = 0 ;
is_u3 = 0 ;
/* Locate core99 Uni-N */
uninorth_node = of_find_node_by_name ( NULL , " uni-n " ) ;
/* Locate G5 u3 */
if ( uninorth_node = = NULL ) {
is_u3 = 1 ;
uninorth_node = of_find_node_by_name ( NULL , " u3 " ) ;
}
if ( uninorth_node ) {
2007-05-01 13:54:02 +10:00
const int * revprop = of_get_property ( uninorth_node ,
2006-07-12 15:40:40 +10:00
" device-rev " , NULL ) ;
2005-04-16 15:20:36 -07:00
if ( revprop ! = NULL )
uninorth_rev = * revprop & 0x3f ;
of_node_put ( uninorth_node ) ;
}
2005-04-16 15:24:19 -07:00
# ifdef CONFIG_PM
/* Inform platform of our suspend/resume caps */
pmac_register_agp_pm ( pdev , agp_uninorth_suspend , agp_uninorth_resume ) ;
# endif
/* Allocate & setup our driver */
2005-04-16 15:20:36 -07:00
bridge = agp_alloc_bridge ( ) ;
if ( ! bridge )
return - ENOMEM ;
if ( is_u3 )
bridge - > driver = & u3_agp_driver ;
else
bridge - > driver = & uninorth_agp_driver ;
bridge - > dev = pdev ;
bridge - > capndx = cap_ptr ;
bridge - > flags = AGP_ERRATA_FASTWRITES ;
/* Fill in the mode register */
pci_read_config_dword ( pdev , cap_ptr + PCI_AGP_STATUS , & bridge - > mode ) ;
pci_set_drvdata ( pdev , bridge ) ;
return agp_add_bridge ( bridge ) ;
}
static void __devexit agp_uninorth_remove ( struct pci_dev * pdev )
{
struct agp_bridge_data * bridge = pci_get_drvdata ( pdev ) ;
2005-04-16 15:24:19 -07:00
# ifdef CONFIG_PM
/* Inform platform of our suspend/resume caps */
pmac_register_agp_pm ( pdev , NULL , NULL ) ;
# endif
2005-04-16 15:20:36 -07:00
agp_remove_bridge ( bridge ) ;
agp_put_bridge ( bridge ) ;
}
static struct pci_device_id agp_uninorth_pci_table [ ] = {
{
. class = ( PCI_CLASS_BRIDGE_HOST < < 8 ) ,
. class_mask = ~ 0 ,
. vendor = PCI_VENDOR_ID_APPLE ,
. device = PCI_ANY_ID ,
. subvendor = PCI_ANY_ID ,
. subdevice = PCI_ANY_ID ,
} ,
{ }
} ;
MODULE_DEVICE_TABLE ( pci , agp_uninorth_pci_table ) ;
static struct pci_driver agp_uninorth_pci_driver = {
. name = " agpgart-uninorth " ,
. id_table = agp_uninorth_pci_table ,
. probe = agp_uninorth_probe ,
. remove = agp_uninorth_remove ,
} ;
static int __init agp_uninorth_init ( void )
{
if ( agp_off )
return - EINVAL ;
return pci_register_driver ( & agp_uninorth_pci_driver ) ;
}
static void __exit agp_uninorth_cleanup ( void )
{
pci_unregister_driver ( & agp_uninorth_pci_driver ) ;
}
module_init ( agp_uninorth_init ) ;
module_exit ( agp_uninorth_cleanup ) ;
2006-10-04 14:56:44 +02:00
module_param ( aperture , charp , 0 ) ;
MODULE_PARM_DESC ( aperture ,
" Aperture size, must be power of two between 4MB and an \n "
" \t \t upper limit specific to the UniNorth revision. \n "
" \t \t Default: 32M " ) ;
2005-04-16 15:20:36 -07:00
MODULE_AUTHOR ( " Ben Herrenschmidt & Paul Mackerras " ) ;
MODULE_LICENSE ( " GPL " ) ;