2005-04-16 15:20:36 -07:00
/* Prom access routines for the sun3x */
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/tty.h>
# include <linux/console.h>
# include <linux/init.h>
# include <linux/mm.h>
# include <linux/string.h>
# include <asm/page.h>
# include <asm/pgtable.h>
# include <asm/bootinfo.h>
# include <asm/setup.h>
# include <asm/traps.h>
# include <asm/sun3xprom.h>
# include <asm/idprom.h>
# include <asm/segment.h>
# include <asm/sun3ints.h>
# include <asm/openprom.h>
# include <asm/machines.h>
void ( * sun3x_putchar ) ( int ) ;
int ( * sun3x_getchar ) ( void ) ;
int ( * sun3x_mayget ) ( void ) ;
int ( * sun3x_mayput ) ( int ) ;
void ( * sun3x_prom_reboot ) ( void ) ;
e_vector sun3x_prom_abort ;
struct linux_romvec * romvec ;
/* prom vector table */
e_vector * sun3x_prom_vbr ;
/* Handle returning to the prom */
void sun3x_halt ( void )
{
2007-05-01 22:32:43 +02:00
unsigned long flags ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Disable interrupts while we mess with things */
local_irq_save ( flags ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Restore prom vbr */
asm volatile ( " movec %0,%%vbr " : : " r " ( ( void * ) sun3x_prom_vbr ) ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Restore prom NMI clock */
// sun3x_disable_intreg(5);
sun3_enable_irq ( 7 ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Let 'er rip */
asm volatile ( " trap #14 " ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Restore everything */
sun3_disable_irq ( 7 ) ;
sun3_enable_irq ( 5 ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
asm volatile ( " movec %0,%%vbr " : : " r " ( ( void * ) vectors ) ) ;
local_irq_restore ( flags ) ;
2005-04-16 15:20:36 -07:00
}
void sun3x_reboot ( void )
{
2007-05-01 22:32:43 +02:00
/* This never returns, don't bother saving things */
local_irq_disable ( ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Restore prom vbr */
asm volatile ( " movec %0,%%vbr " : : " r " ( ( void * ) sun3x_prom_vbr ) ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Restore prom NMI clock */
sun3_disable_irq ( 5 ) ;
sun3_enable_irq ( 7 ) ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Let 'er rip */
( * romvec - > pv_reboot ) ( " vmlinux " ) ;
2005-04-16 15:20:36 -07:00
}
static void sun3x_prom_write ( struct console * co , const char * s ,
unsigned int count )
{
2007-05-01 22:32:43 +02:00
while ( count - - ) {
if ( * s = = ' \n ' )
sun3x_putchar ( ' \r ' ) ;
sun3x_putchar ( * s + + ) ;
}
2005-04-16 15:20:36 -07:00
}
/* debug console - write-only */
static struct console sun3x_debug = {
2007-05-01 22:32:43 +02:00
. name = " debug " ,
. write = sun3x_prom_write ,
. flags = CON_PRINTBUFFER ,
. index = - 1 ,
2005-04-16 15:20:36 -07:00
} ;
2007-07-20 04:33:28 +01:00
void __init sun3x_prom_init ( void )
2005-04-16 15:20:36 -07:00
{
2007-05-01 22:32:43 +02:00
/* Read the vector table */
sun3x_putchar = * ( void ( * * ) ( int ) ) ( SUN3X_P_PUTCHAR ) ;
sun3x_getchar = * ( int ( * * ) ( void ) ) ( SUN3X_P_GETCHAR ) ;
sun3x_mayget = * ( int ( * * ) ( void ) ) ( SUN3X_P_MAYGET ) ;
sun3x_mayput = * ( int ( * * ) ( int ) ) ( SUN3X_P_MAYPUT ) ;
sun3x_prom_reboot = * ( void ( * * ) ( void ) ) ( SUN3X_P_REBOOT ) ;
sun3x_prom_abort = * ( e_vector * ) ( SUN3X_P_ABORT ) ;
romvec = ( struct linux_romvec * ) SUN3X_PROM_BASE ;
idprom_init ( ) ;
if ( ! ( ( idprom - > id_machtype & SM_ARCH_MASK ) = = SM_SUN3X ) ) {
printk ( " Warning: machine reports strange type %02x \n " ,
idprom - > id_machtype ) ;
printk ( " Pretending it's a 3/80, but very afraid... \n " ) ;
idprom - > id_machtype = SM_SUN3X | SM_3_80 ;
}
/* point trap #14 at abort.
* XXX this is futile since we restore the vbr first - oops
*/
vectors [ VEC_TRAP14 ] = sun3x_prom_abort ;
2007-05-01 22:32:45 +02:00
}
2007-05-01 22:32:43 +02:00
2007-05-01 22:32:45 +02:00
static int __init sun3x_debug_setup ( char * arg )
{
2007-05-01 22:32:43 +02:00
/* If debug=prom was specified, start the debug console */
2007-05-01 22:32:45 +02:00
if ( MACH_IS_SUN3X & & ! strcmp ( arg , " prom " ) )
2007-05-01 22:32:43 +02:00
register_console ( & sun3x_debug ) ;
2007-05-01 22:32:45 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-05-01 22:32:45 +02:00
early_param ( " debug " , sun3x_debug_setup ) ;
2005-04-16 15:20:36 -07:00
/* some prom functions to export */
int prom_getintdefault ( int node , char * property , int deflt )
{
return deflt ;
}
int prom_getbool ( int node , char * prop )
{
return 1 ;
}
void prom_printf ( char * fmt , . . . )
{
}
void prom_halt ( void )
{
sun3x_halt ( ) ;
}
/* Get the idprom and stuff it into buffer 'idbuf'. Returns the
* format type . ' num_bytes ' is the number of bytes that your idbuf
* has space for . Returns 0xff on error .
*/
unsigned char
prom_get_idprom ( char * idbuf , int num_bytes )
{
int i ;
/* make a copy of the idprom structure */
2007-05-01 22:32:43 +02:00
for ( i = 0 ; i < num_bytes ; i + + )
2005-04-16 15:20:36 -07:00
idbuf [ i ] = ( ( char * ) SUN3X_IDPROM ) [ i ] ;
return idbuf [ 0 ] ;
}