2005-04-16 15:20:36 -07:00
/*
* linux / arch / m68k / amiga / config . c
*
* Copyright ( C ) 1993 Hamish Macdonald
*
* 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 .
*/
/*
* Miscellaneous Amiga stuff
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/mm.h>
2008-10-03 22:42:36 +04:00
# include <linux/seq_file.h>
2005-04-16 15:20:36 -07:00
# include <linux/tty.h>
2018-12-01 11:53:10 +11:00
# include <linux/clocksource.h>
2005-04-16 15:20:36 -07:00
# include <linux/console.h>
# include <linux/rtc.h>
# include <linux/init.h>
# include <linux/vt_kern.h>
# include <linux/delay.h>
# include <linux/interrupt.h>
# include <linux/zorro.h>
2008-02-04 22:30:25 -08:00
# include <linux/module.h>
2008-10-13 21:58:53 +02:00
# include <linux/keyboard.h>
2005-04-16 15:20:36 -07:00
# include <asm/bootinfo.h>
2013-10-02 11:37:33 +02:00
# include <asm/bootinfo-amiga.h>
2013-10-04 09:38:53 +02:00
# include <asm/byteorder.h>
2005-04-16 15:20:36 -07:00
# include <asm/setup.h>
# include <asm/amigahw.h>
# include <asm/amigaints.h>
# include <asm/irq.h>
# include <asm/machdep.h>
# include <asm/io.h>
2022-01-21 21:07:34 +01:00
# include <asm/config.h>
2005-04-16 15:20:36 -07:00
2023-09-13 16:08:07 +02:00
# include "amiga.h"
2008-07-17 21:16:23 +02:00
static unsigned long amiga_model ;
2008-02-04 22:30:25 -08:00
2005-04-16 15:20:36 -07:00
unsigned long amiga_eclock ;
2008-02-04 22:30:25 -08:00
EXPORT_SYMBOL ( amiga_eclock ) ;
2005-04-16 15:20:36 -07:00
unsigned long amiga_colorclock ;
2008-02-04 22:30:25 -08:00
EXPORT_SYMBOL ( amiga_colorclock ) ;
2005-04-16 15:20:36 -07:00
unsigned long amiga_chipset ;
2008-02-04 22:30:25 -08:00
EXPORT_SYMBOL ( amiga_chipset ) ;
2005-04-16 15:20:36 -07:00
unsigned char amiga_vblank ;
2008-07-17 21:16:21 +02:00
EXPORT_SYMBOL ( amiga_vblank ) ;
2008-07-17 21:16:23 +02:00
static unsigned char amiga_psfreq ;
2008-02-04 22:30:25 -08:00
2005-04-16 15:20:36 -07:00
struct amiga_hw_present amiga_hw_present ;
2008-02-04 22:30:25 -08:00
EXPORT_SYMBOL ( amiga_hw_present ) ;
2005-04-16 15:20:36 -07:00
static char s_a500 [ ] __initdata = " A500 " ;
static char s_a500p [ ] __initdata = " A500+ " ;
static char s_a600 [ ] __initdata = " A600 " ;
static char s_a1000 [ ] __initdata = " A1000 " ;
static char s_a1200 [ ] __initdata = " A1200 " ;
static char s_a2000 [ ] __initdata = " A2000 " ;
static char s_a2500 [ ] __initdata = " A2500 " ;
static char s_a3000 [ ] __initdata = " A3000 " ;
static char s_a3000t [ ] __initdata = " A3000T " ;
static char s_a3000p [ ] __initdata = " A3000+ " ;
static char s_a4000 [ ] __initdata = " A4000 " ;
static char s_a4000t [ ] __initdata = " A4000T " ;
static char s_cdtv [ ] __initdata = " CDTV " ;
static char s_cd32 [ ] __initdata = " CD32 " ;
static char s_draco [ ] __initdata = " Draco " ;
static char * amiga_models [ ] __initdata = {
2007-05-01 22:32:43 +02:00
[ AMI_500 - AMI_500 ] = s_a500 ,
[ AMI_500PLUS - AMI_500 ] = s_a500p ,
[ AMI_600 - AMI_500 ] = s_a600 ,
[ AMI_1000 - AMI_500 ] = s_a1000 ,
[ AMI_1200 - AMI_500 ] = s_a1200 ,
[ AMI_2000 - AMI_500 ] = s_a2000 ,
[ AMI_2500 - AMI_500 ] = s_a2500 ,
[ AMI_3000 - AMI_500 ] = s_a3000 ,
[ AMI_3000T - AMI_500 ] = s_a3000t ,
[ AMI_3000PLUS - AMI_500 ] = s_a3000p ,
[ AMI_4000 - AMI_500 ] = s_a4000 ,
[ AMI_4000T - AMI_500 ] = s_a4000t ,
[ AMI_CDTV - AMI_500 ] = s_cdtv ,
[ AMI_CD32 - AMI_500 ] = s_cd32 ,
[ AMI_DRACO - AMI_500 ] = s_draco ,
2005-04-16 15:20:36 -07:00
} ;
static char amiga_model_name [ 13 ] = " Amiga " ;
2020-09-24 17:29:17 +02:00
static void amiga_sched_init ( void ) ;
2005-04-16 15:20:36 -07:00
static void amiga_get_model ( char * model ) ;
2008-10-03 22:42:36 +04:00
static void amiga_get_hardware_list ( struct seq_file * m ) ;
2007-05-01 22:32:43 +02:00
static void amiga_reset ( void ) ;
2005-04-16 15:20:36 -07:00
static void amiga_mem_console_write ( struct console * co , const char * b ,
unsigned int count ) ;
# ifdef CONFIG_HEARTBEAT
static void amiga_heartbeat ( int on ) ;
# endif
static struct console amiga_console_driver = {
2007-05-01 22:32:43 +02:00
. name = " debug " ,
. flags = CON_PRINTBUFFER ,
. index = - 1 ,
2005-04-16 15:20:36 -07:00
} ;
/*
* Motherboard Resources present in all Amiga models
*/
static struct {
2007-05-01 22:32:43 +02:00
struct resource _ciab , _ciaa , _custom , _kickstart ;
2005-04-16 15:20:36 -07:00
} mb_resources = {
2007-05-01 22:32:43 +02:00
. _ciab = {
. name = " CIA B " , . start = 0x00bfd000 , . end = 0x00bfdfff
} ,
. _ciaa = {
. name = " CIA A " , . start = 0x00bfe000 , . end = 0x00bfefff
} ,
. _custom = {
. name = " Custom I/O " , . start = 0x00dff000 , . end = 0x00dfffff
} ,
. _kickstart = {
. name = " Kickstart ROM " , . start = 0x00f80000 , . end = 0x00ffffff
}
2005-04-16 15:20:36 -07:00
} ;
static struct resource ram_resource [ NUM_MEMINFO ] ;
/*
* Parse an Amiga - specific record in the bootinfo
*/
2013-06-25 21:15:24 +02:00
int __init amiga_parse_bootinfo ( const struct bi_record * record )
2005-04-16 15:20:36 -07:00
{
2007-05-01 22:32:43 +02:00
int unknown = 0 ;
2013-10-04 11:41:24 +02:00
const void * data = record - > data ;
2005-04-16 15:20:36 -07:00
2013-10-04 11:41:24 +02:00
switch ( be16_to_cpu ( record - > tag ) ) {
2005-04-16 15:20:36 -07:00
case BI_AMIGA_MODEL :
2013-10-04 11:41:24 +02:00
amiga_model = be32_to_cpup ( data ) ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_ECLOCK :
2013-10-04 11:41:24 +02:00
amiga_eclock = be32_to_cpup ( data ) ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_CHIPSET :
2013-10-04 11:41:24 +02:00
amiga_chipset = be32_to_cpup ( data ) ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_CHIP_SIZE :
2013-10-04 11:41:24 +02:00
amiga_chip_size = be32_to_cpup ( data ) ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_VBLANK :
2013-10-04 11:41:24 +02:00
amiga_vblank = * ( const __u8 * ) data ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_PSFREQ :
2013-10-04 11:41:24 +02:00
amiga_psfreq = * ( const __u8 * ) data ;
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_AUTOCON :
# ifdef CONFIG_ZORRO
2007-05-01 22:32:43 +02:00
if ( zorro_num_autocon < ZORRO_NUM_AUTO ) {
2013-10-04 11:41:24 +02:00
const struct ConfigDev * cd = data ;
2013-06-29 10:40:20 +02:00
struct zorro_dev_init * dev = & zorro_autocon_init [ zorro_num_autocon + + ] ;
2007-05-01 22:32:43 +02:00
dev - > rom = cd - > cd_Rom ;
2013-10-04 09:38:53 +02:00
dev - > slotaddr = be16_to_cpu ( cd - > cd_SlotAddr ) ;
dev - > slotsize = be16_to_cpu ( cd - > cd_SlotSize ) ;
dev - > boardaddr = be32_to_cpu ( cd - > cd_BoardAddr ) ;
dev - > boardsize = be32_to_cpu ( cd - > cd_BoardSize ) ;
2007-05-01 22:32:43 +02:00
} else
2014-05-10 12:38:00 +02:00
pr_warn ( " amiga_parse_bootinfo: too many AutoConfig devices \n " ) ;
2005-04-16 15:20:36 -07:00
# endif /* CONFIG_ZORRO */
2007-05-01 22:32:43 +02:00
break ;
2005-04-16 15:20:36 -07:00
case BI_AMIGA_SERPER :
2007-05-01 22:32:43 +02:00
/* serial port period: ignored here */
break ;
2005-04-16 15:20:36 -07:00
default :
2007-05-01 22:32:43 +02:00
unknown = 1 ;
}
return unknown ;
2005-04-16 15:20:36 -07:00
}
/*
* Identify builtin hardware
*/
static void __init amiga_identify ( void )
{
2007-05-01 22:32:43 +02:00
/* Fill in some default values, if necessary */
if ( amiga_eclock = = 0 )
amiga_eclock = 709379 ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
memset ( & amiga_hw_present , 0 , sizeof ( amiga_hw_present ) ) ;
2014-05-10 12:38:00 +02:00
pr_info ( " Amiga hardware found: " ) ;
2007-05-01 22:32:43 +02:00
if ( amiga_model > = AMI_500 & & amiga_model < = AMI_DRACO ) {
2014-05-10 12:38:00 +02:00
pr_cont ( " [%s] " , amiga_models [ amiga_model - AMI_500 ] ) ;
2007-05-01 22:32:43 +02:00
strcat ( amiga_model_name , amiga_models [ amiga_model - AMI_500 ] ) ;
}
switch ( amiga_model ) {
case AMI_UNKNOWN :
2020-08-26 14:51:24 +02:00
break ;
2007-05-01 22:32:43 +02:00
case AMI_600 :
case AMI_1200 :
AMIGAHW_SET ( A1200_IDE ) ;
AMIGAHW_SET ( PCMCIA ) ;
2020-03-26 18:29:09 -05:00
fallthrough ;
2007-05-01 22:32:43 +02:00
case AMI_500 :
case AMI_500PLUS :
case AMI_1000 :
case AMI_2000 :
case AMI_2500 :
AMIGAHW_SET ( A2000_CLK ) ; /* Is this correct for all models? */
2020-08-26 14:51:24 +02:00
break ;
2007-05-01 22:32:43 +02:00
case AMI_3000 :
case AMI_3000T :
AMIGAHW_SET ( AMBER_FF ) ;
AMIGAHW_SET ( MAGIC_REKICK ) ;
2020-03-26 18:29:09 -05:00
fallthrough ;
2007-05-01 22:32:43 +02:00
case AMI_3000PLUS :
AMIGAHW_SET ( A3000_SCSI ) ;
AMIGAHW_SET ( A3000_CLK ) ;
AMIGAHW_SET ( ZORRO3 ) ;
2020-08-26 14:51:24 +02:00
break ;
2007-05-01 22:32:43 +02:00
case AMI_4000T :
AMIGAHW_SET ( A4000_SCSI ) ;
2020-03-26 18:29:09 -05:00
fallthrough ;
2007-05-01 22:32:43 +02:00
case AMI_4000 :
AMIGAHW_SET ( A4000_IDE ) ;
AMIGAHW_SET ( A3000_CLK ) ;
AMIGAHW_SET ( ZORRO3 ) ;
2020-08-26 14:51:24 +02:00
break ;
2007-05-01 22:32:43 +02:00
case AMI_CDTV :
case AMI_CD32 :
AMIGAHW_SET ( CD_ROM ) ;
AMIGAHW_SET ( A2000_CLK ) ; /* Is this correct? */
break ;
case AMI_DRACO :
panic ( " No support for Draco yet " ) ;
default :
panic ( " Unknown Amiga Model " ) ;
}
2020-08-26 14:51:24 +02:00
AMIGAHW_SET ( AMI_VIDEO ) ;
AMIGAHW_SET ( AMI_BLITTER ) ;
AMIGAHW_SET ( AMI_AUDIO ) ;
AMIGAHW_SET ( AMI_FLOPPY ) ;
AMIGAHW_SET ( AMI_KEYBOARD ) ;
AMIGAHW_SET ( AMI_MOUSE ) ;
AMIGAHW_SET ( AMI_SERIAL ) ;
AMIGAHW_SET ( AMI_PARALLEL ) ;
AMIGAHW_SET ( CHIP_RAM ) ;
AMIGAHW_SET ( PAULA ) ;
switch ( amiga_chipset ) {
case CS_OCS :
case CS_ECS :
case CS_AGA :
switch ( amiga_custom . deniseid & 0xf ) {
case 0x0c :
AMIGAHW_SET ( DENISE_HR ) ;
break ;
case 0x08 :
AMIGAHW_SET ( LISA ) ;
break ;
default :
AMIGAHW_SET ( DENISE ) ;
break ;
}
break ;
}
switch ( ( amiga_custom . vposr > > 8 ) & 0x7f ) {
case 0x00 :
AMIGAHW_SET ( AGNUS_PAL ) ;
break ;
case 0x10 :
AMIGAHW_SET ( AGNUS_NTSC ) ;
break ;
case 0x20 :
case 0x21 :
AMIGAHW_SET ( AGNUS_HR_PAL ) ;
break ;
case 0x30 :
case 0x31 :
AMIGAHW_SET ( AGNUS_HR_NTSC ) ;
break ;
case 0x22 :
case 0x23 :
AMIGAHW_SET ( ALICE_PAL ) ;
break ;
case 0x32 :
case 0x33 :
AMIGAHW_SET ( ALICE_NTSC ) ;
break ;
}
AMIGAHW_SET ( ZORRO ) ;
2007-05-01 22:32:43 +02:00
# define AMIGAHW_ANNOUNCE(name, str) \
if ( AMIGAHW_PRESENT ( name ) ) \
2014-05-10 12:38:00 +02:00
pr_cont ( str )
2007-05-01 22:32:43 +02:00
AMIGAHW_ANNOUNCE ( AMI_VIDEO , " VIDEO " ) ;
AMIGAHW_ANNOUNCE ( AMI_BLITTER , " BLITTER " ) ;
AMIGAHW_ANNOUNCE ( AMBER_FF , " AMBER_FF " ) ;
AMIGAHW_ANNOUNCE ( AMI_AUDIO , " AUDIO " ) ;
AMIGAHW_ANNOUNCE ( AMI_FLOPPY , " FLOPPY " ) ;
AMIGAHW_ANNOUNCE ( A3000_SCSI , " A3000_SCSI " ) ;
AMIGAHW_ANNOUNCE ( A4000_SCSI , " A4000_SCSI " ) ;
AMIGAHW_ANNOUNCE ( A1200_IDE , " A1200_IDE " ) ;
AMIGAHW_ANNOUNCE ( A4000_IDE , " A4000_IDE " ) ;
AMIGAHW_ANNOUNCE ( CD_ROM , " CD_ROM " ) ;
AMIGAHW_ANNOUNCE ( AMI_KEYBOARD , " KEYBOARD " ) ;
AMIGAHW_ANNOUNCE ( AMI_MOUSE , " MOUSE " ) ;
AMIGAHW_ANNOUNCE ( AMI_SERIAL , " SERIAL " ) ;
AMIGAHW_ANNOUNCE ( AMI_PARALLEL , " PARALLEL " ) ;
AMIGAHW_ANNOUNCE ( A2000_CLK , " A2000_CLK " ) ;
AMIGAHW_ANNOUNCE ( A3000_CLK , " A3000_CLK " ) ;
AMIGAHW_ANNOUNCE ( CHIP_RAM , " CHIP_RAM " ) ;
AMIGAHW_ANNOUNCE ( PAULA , " PAULA " ) ;
AMIGAHW_ANNOUNCE ( DENISE , " DENISE " ) ;
AMIGAHW_ANNOUNCE ( DENISE_HR , " DENISE_HR " ) ;
AMIGAHW_ANNOUNCE ( LISA , " LISA " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_PAL , " AGNUS_PAL " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_NTSC , " AGNUS_NTSC " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_HR_PAL , " AGNUS_HR_PAL " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_HR_NTSC , " AGNUS_HR_NTSC " ) ;
AMIGAHW_ANNOUNCE ( ALICE_PAL , " ALICE_PAL " ) ;
AMIGAHW_ANNOUNCE ( ALICE_NTSC , " ALICE_NTSC " ) ;
AMIGAHW_ANNOUNCE ( MAGIC_REKICK , " MAGIC_REKICK " ) ;
AMIGAHW_ANNOUNCE ( PCMCIA , " PCMCIA " ) ;
if ( AMIGAHW_PRESENT ( ZORRO ) )
2014-05-10 12:38:00 +02:00
pr_cont ( " ZORRO%s " , AMIGAHW_PRESENT ( ZORRO3 ) ? " 3 " : " " ) ;
pr_cont ( " \n " ) ;
2005-04-16 15:20:36 -07:00
# undef AMIGAHW_ANNOUNCE
}
2013-10-18 13:16:06 +02:00
static unsigned long amiga_random_get_entropy ( void )
{
/* VPOSR/VHPOSR provide at least 17 bits of data changing at 1.79 MHz */
return * ( unsigned long * ) & amiga_custom . vposr ;
}
2005-04-16 15:20:36 -07:00
/*
* Setup the Amiga configuration info
*/
void __init config_amiga ( void )
{
2007-05-01 22:32:43 +02:00
int i ;
amiga_identify ( ) ;
/* Yuk, we don't have PCI memory */
iomem_resource . name = " Memory " ;
for ( i = 0 ; i < 4 ; i + + )
request_resource ( & iomem_resource , & ( ( struct resource * ) & mb_resources ) [ i ] ) ;
mach_sched_init = amiga_sched_init ;
mach_init_IRQ = amiga_init_IRQ ;
mach_get_model = amiga_get_model ;
mach_get_hardware_list = amiga_get_hardware_list ;
mach_reset = amiga_reset ;
2016-09-05 13:18:15 +02:00
# if IS_ENABLED(CONFIG_INPUT_M68K_BEEP)
2007-05-01 22:32:43 +02:00
mach_beep = amiga_mksound ;
2005-04-16 15:20:36 -07:00
# endif
# ifdef CONFIG_HEARTBEAT
2007-05-01 22:32:43 +02:00
mach_heartbeat = amiga_heartbeat ;
2005-04-16 15:20:36 -07:00
# endif
2013-10-18 13:16:06 +02:00
mach_random_get_entropy = amiga_random_get_entropy ;
2008-07-17 21:16:23 +02:00
/* Fill in the clock value (based on the 700 kHz E-Clock) */
2007-05-01 22:32:43 +02:00
amiga_colorclock = 5 * amiga_eclock ; /* 3.5 MHz */
/* clear all DMA bits */
amiga_custom . dmacon = DMAF_ALL ;
/* ensure that the DMA master bit is set */
amiga_custom . dmacon = DMAF_SETCLR | DMAF_MASTER ;
/* don't use Z2 RAM as system memory on Z3 capable machines */
if ( AMIGAHW_PRESENT ( ZORRO3 ) ) {
int i , j ;
u32 disabled_z2mem = 0 ;
for ( i = 0 ; i < m68k_num_memory ; i + + ) {
if ( m68k_memory [ i ] . addr < 16 * 1024 * 1024 ) {
if ( i = = 0 ) {
/* don't cut off the branch we're sitting on */
2014-05-10 12:38:00 +02:00
pr_warn ( " Warning: kernel runs in Zorro II memory \n " ) ;
2007-05-01 22:32:43 +02:00
continue ;
}
disabled_z2mem + = m68k_memory [ i ] . size ;
m68k_num_memory - - ;
for ( j = i ; j < m68k_num_memory ; j + + )
m68k_memory [ j ] = m68k_memory [ j + 1 ] ;
i - - ;
}
}
if ( disabled_z2mem )
2014-05-10 12:38:00 +02:00
pr_info ( " %dK of Zorro II memory will not be used as system memory \n " ,
disabled_z2mem > > 10 ) ;
2007-05-01 22:32:43 +02:00
}
/* request all RAM */
for ( i = 0 ; i < m68k_num_memory ; i + + ) {
ram_resource [ i ] . name =
( m68k_memory [ i ] . addr > = 0x01000000 ) ? " 32-bit Fast RAM " :
( m68k_memory [ i ] . addr < 0x00c00000 ) ? " 16-bit Fast RAM " :
" 16-bit Slow RAM " ;
ram_resource [ i ] . start = m68k_memory [ i ] . addr ;
ram_resource [ i ] . end = m68k_memory [ i ] . addr + m68k_memory [ i ] . size - 1 ;
request_resource ( & iomem_resource , & ram_resource [ i ] ) ;
}
/* initialize chipram allocator */
amiga_chip_init ( ) ;
/* our beloved beeper */
if ( AMIGAHW_PRESENT ( AMI_AUDIO ) )
amiga_init_sound ( ) ;
/*
* if it is an A3000 , set the magic bit that forces
* a hard rekick
*/
if ( AMIGAHW_PRESENT ( MAGIC_REKICK ) )
* ( unsigned char * ) ZTWO_VADDR ( 0xde0002 ) | = 0x80 ;
2005-04-16 15:20:36 -07:00
}
2018-12-01 11:53:10 +11:00
static u64 amiga_read_clk ( struct clocksource * cs ) ;
static struct clocksource amiga_clk = {
. name = " ciab " ,
. rating = 250 ,
. read = amiga_read_clk ,
. mask = CLOCKSOURCE_MASK ( 32 ) ,
. flags = CLOCK_SOURCE_IS_CONTINUOUS ,
} ;
2005-04-16 15:20:36 -07:00
static unsigned short jiffy_ticks ;
2018-12-01 11:53:10 +11:00
static u32 clk_total , clk_offset ;
static irqreturn_t ciab_timer_handler ( int irq , void * dev_id )
{
clk_total + = jiffy_ticks ;
clk_offset = 0 ;
2020-09-24 17:21:00 +02:00
legacy_timer_tick ( 1 ) ;
2020-09-24 16:37:37 +02:00
timer_heartbeat ( ) ;
2018-12-01 11:53:10 +11:00
return IRQ_HANDLED ;
}
2005-04-16 15:20:36 -07:00
2020-09-24 17:29:17 +02:00
static void __init amiga_sched_init ( void )
2005-04-16 15:20:36 -07:00
{
static struct resource sched_res = {
2007-05-01 22:32:43 +02:00
. name = " timer " , . start = 0x00bfd400 , . end = 0x00bfd5ff ,
2005-04-16 15:20:36 -07:00
} ;
2009-08-02 10:47:47 +02:00
jiffy_ticks = DIV_ROUND_CLOSEST ( amiga_eclock , HZ ) ;
2005-04-16 15:20:36 -07:00
if ( request_resource ( & mb_resources . _ciab , & sched_res ) )
2014-05-10 12:38:00 +02:00
pr_warn ( " Cannot allocate ciab.ta{lo,hi} \n " ) ;
2005-04-16 15:20:36 -07:00
ciab . cra & = 0xC0 ; /* turn off timer A, continuous mode, from Eclk */
ciab . talo = jiffy_ticks % 256 ;
ciab . tahi = jiffy_ticks / 256 ;
/* install interrupt service routine for CIAB Timer A
*
* Please don ' t change this to use ciaa , as it interferes with the
* SCSI code . We ' ll have to take a look at this later
*/
2018-12-01 11:53:10 +11:00
if ( request_irq ( IRQ_AMIGA_CIAB_TA , ciab_timer_handler , IRQF_TIMER ,
2020-09-24 17:21:00 +02:00
" timer " , NULL ) )
2008-12-30 14:00:34 +01:00
pr_err ( " Couldn't register timer interrupt \n " ) ;
2005-04-16 15:20:36 -07:00
/* start timer */
ciab . cra | = 0x11 ;
2018-12-01 11:53:10 +11:00
clocksource_register_hz ( & amiga_clk , amiga_eclock ) ;
}
2005-04-16 15:20:36 -07:00
2018-12-01 11:53:10 +11:00
static u64 amiga_read_clk ( struct clocksource * cs )
2005-04-16 15:20:36 -07:00
{
unsigned short hi , lo , hi2 ;
2018-12-01 11:53:10 +11:00
unsigned long flags ;
u32 ticks ;
local_irq_save ( flags ) ;
2005-04-16 15:20:36 -07:00
/* read CIA B timer A current value */
hi = ciab . tahi ;
lo = ciab . talo ;
hi2 = ciab . tahi ;
if ( hi ! = hi2 ) {
lo = ciab . talo ;
hi = hi2 ;
}
ticks = hi < < 8 | lo ;
if ( ticks > jiffy_ticks / 2 )
/* check for pending interrupt */
if ( cia_set_irq ( & ciab_base , 0 ) & CIA_ICR_TA )
2018-12-01 11:53:10 +11:00
clk_offset = jiffy_ticks ;
2005-04-16 15:20:36 -07:00
ticks = jiffy_ticks - ticks ;
2018-12-01 11:53:10 +11:00
ticks + = clk_offset + clk_total ;
local_irq_restore ( flags ) ;
2005-04-16 15:20:36 -07:00
2018-12-01 11:53:10 +11:00
return ticks ;
2005-04-16 15:20:36 -07:00
}
2012-01-12 17:17:21 -08:00
static void amiga_reset ( void ) __noreturn ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
static void amiga_reset ( void )
2005-04-16 15:20:36 -07:00
{
2007-05-01 22:32:43 +02:00
unsigned long jmp_addr040 = virt_to_phys ( & & jmp_addr_label040 ) ;
unsigned long jmp_addr = virt_to_phys ( & & jmp_addr_label ) ;
local_irq_disable ( ) ;
if ( CPU_IS_040_OR_060 )
/* Setup transparent translation registers for mapping
* of 16 MB kernel segment before disabling translation
*/
asm volatile ( " \n "
" move.l %0,%%d0 \n "
" and.l #0xff000000,%%d0 \n "
" or.w #0xe020,%%d0 \n " /* map 16 MB, enable, cacheable */
" .chip 68040 \n "
" movec %%d0,%%itt0 \n "
" movec %%d0,%%dtt0 \n "
" .chip 68k \n "
" jmp %0@ \n "
: /* no outputs */
: " a " ( jmp_addr040 )
: " d0 " ) ;
else
/* for 680[23]0, just disable translation and jump to the physical
* address of the label
*/
asm volatile ( " \n "
" pmove %%tc,%@ \n "
" bclr #7,%@ \n "
" pmove %@,%%tc \n "
" jmp %0@ \n "
: /* no outputs */
: " a " ( jmp_addr ) ) ;
jmp_addr_label040 :
/* disable translation on '040 now */
asm volatile ( " \n "
" moveq #0,%%d0 \n "
" .chip 68040 \n "
" movec %%d0,%%tc \n " /* disable MMU */
" .chip 68k \n "
: /* no outputs */
: /* no inputs */
: " d0 " ) ;
jmp_addr_label :
/* pickup reset address from AmigaOS ROM, reset devices and jump
* to reset address
*/
asm volatile ( " \n "
" move.w #0x2700,%sr \n "
" lea 0x01000000,%a0 \n "
" sub.l %a0@(-0x14),%a0 \n "
" move.l %a0@(4),%a0 \n "
" subq.l #2,%a0 \n "
" jra 1f \n "
/* align on a longword boundary */
" " __ALIGN_STR " \n "
" 1: \n "
" reset \n "
" jmp %a0@ " ) ;
for ( ; ; )
;
2005-04-16 15:20:36 -07:00
}
/*
* Debugging
*/
# define SAVEKMSG_MAXMEM 128*1024
# define SAVEKMSG_MAGIC1 0x53415645 /* 'SAVE' */
# define SAVEKMSG_MAGIC2 0x4B4D5347 /* 'KMSG' */
struct savekmsg {
2007-05-01 22:32:43 +02:00
unsigned long magic1 ; /* SAVEKMSG_MAGIC1 */
unsigned long magic2 ; /* SAVEKMSG_MAGIC2 */
unsigned long magicptr ; /* address of magic1 */
unsigned long size ;
2020-04-20 13:14:01 -05:00
char data [ ] ;
2005-04-16 15:20:36 -07:00
} ;
static struct savekmsg * savekmsg ;
static void amiga_mem_console_write ( struct console * co , const char * s ,
unsigned int count )
{
2007-05-01 22:32:43 +02:00
if ( savekmsg - > size + count < = SAVEKMSG_MAXMEM - sizeof ( struct savekmsg ) ) {
memcpy ( savekmsg - > data + savekmsg - > size , s , count ) ;
savekmsg - > size + = count ;
}
2005-04-16 15:20:36 -07:00
}
2007-05-01 22:32:45 +02:00
static int __init amiga_savekmsg_setup ( char * arg )
2005-04-16 15:20:36 -07:00
{
2013-12-01 11:14:51 +01:00
bool registered ;
2007-05-01 22:32:45 +02:00
if ( ! MACH_IS_AMIGA | | strcmp ( arg , " mem " ) )
2011-01-07 13:25:29 +01:00
return 0 ;
2007-05-01 22:32:45 +02:00
2011-01-07 13:25:29 +01:00
if ( amiga_chip_size < SAVEKMSG_MAXMEM ) {
pr_err ( " Not enough chipram for debugging \n " ) ;
return - ENOMEM ;
2007-05-01 22:32:45 +02:00
}
2011-01-07 13:25:29 +01:00
/* Just steal the block, the chipram allocator isn't functional yet */
amiga_chip_size - = SAVEKMSG_MAXMEM ;
2011-01-09 11:03:43 +01:00
savekmsg = ZTWO_VADDR ( CHIP_PHYSADDR + amiga_chip_size ) ;
2007-05-01 22:32:43 +02:00
savekmsg - > magic1 = SAVEKMSG_MAGIC1 ;
savekmsg - > magic2 = SAVEKMSG_MAGIC2 ;
savekmsg - > magicptr = ZTWO_PADDR ( savekmsg ) ;
savekmsg - > size = 0 ;
2007-05-01 22:32:45 +02:00
2013-12-01 11:14:51 +01:00
registered = ! ! amiga_console_driver . write ;
2007-05-01 22:32:45 +02:00
amiga_console_driver . write = amiga_mem_console_write ;
2013-12-01 11:14:51 +01:00
if ( ! registered )
register_console ( & amiga_console_driver ) ;
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 " , amiga_savekmsg_setup ) ;
2005-04-16 15:20:36 -07:00
static void amiga_serial_putc ( char c )
{
2007-05-01 22:32:43 +02:00
amiga_custom . serdat = ( unsigned char ) c | 0x100 ;
while ( ! ( amiga_custom . serdatr & 0x2000 ) )
;
2005-04-16 15:20:36 -07:00
}
2008-07-17 21:16:23 +02:00
static void amiga_serial_console_write ( struct console * co , const char * s ,
unsigned int count )
2005-04-16 15:20:36 -07:00
{
2007-05-01 22:32:43 +02:00
while ( count - - ) {
if ( * s = = ' \n ' )
amiga_serial_putc ( ' \r ' ) ;
amiga_serial_putc ( * s + + ) ;
}
2005-04-16 15:20:36 -07:00
}
2008-07-17 21:16:23 +02:00
#if 0
2005-04-16 15:20:36 -07:00
void amiga_serial_puts ( const char * s )
{
2007-05-01 22:32:43 +02:00
amiga_serial_console_write ( NULL , s , strlen ( s ) ) ;
2005-04-16 15:20:36 -07:00
}
int amiga_serial_console_wait_key ( struct console * co )
{
2007-05-01 22:32:43 +02:00
int ch ;
while ( ! ( amiga_custom . intreqr & IF_RBF ) )
barrier ( ) ;
ch = amiga_custom . serdatr & 0xff ;
/* clear the interrupt, so that another character can be read */
amiga_custom . intreq = IF_RBF ;
return ch ;
2005-04-16 15:20:36 -07:00
}
void amiga_serial_gets ( struct console * co , char * s , int len )
{
2007-05-01 22:32:43 +02:00
int ch , cnt = 0 ;
while ( 1 ) {
ch = amiga_serial_console_wait_key ( co ) ;
/* Check for backspace. */
if ( ch = = 8 | | ch = = 127 ) {
if ( cnt = = 0 ) {
amiga_serial_putc ( ' \007 ' ) ;
continue ;
}
cnt - - ;
amiga_serial_puts ( " \010 \010 " ) ;
continue ;
}
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Check for enter. */
if ( ch = = 10 | | ch = = 13 )
break ;
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* See if line is too long. */
if ( cnt > = len + 1 ) {
amiga_serial_putc ( 7 ) ;
cnt - - ;
continue ;
}
2005-04-16 15:20:36 -07:00
2007-05-01 22:32:43 +02:00
/* Store and echo character. */
s [ cnt + + ] = ch ;
amiga_serial_putc ( ch ) ;
}
/* Print enter. */
amiga_serial_puts ( " \r \n " ) ;
s [ cnt ] = 0 ;
2005-04-16 15:20:36 -07:00
}
# endif
2007-05-01 22:32:45 +02:00
static int __init amiga_debug_setup ( char * arg )
2005-04-16 15:20:36 -07:00
{
2013-12-01 11:14:51 +01:00
bool registered ;
if ( ! MACH_IS_AMIGA | | strcmp ( arg , " ser " ) )
return 0 ;
/* no initialization required (?) */
registered = ! ! amiga_console_driver . write ;
amiga_console_driver . write = amiga_serial_console_write ;
if ( ! registered )
2005-04-16 15:20:36 -07:00
register_console ( & amiga_console_driver ) ;
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 " , amiga_debug_setup ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_HEARTBEAT
static void amiga_heartbeat ( int on )
{
2007-05-01 22:32:43 +02:00
if ( on )
ciaa . pra & = ~ 2 ;
else
ciaa . pra | = 2 ;
2005-04-16 15:20:36 -07:00
}
# endif
/*
* Amiga specific parts of / proc
*/
static void amiga_get_model ( char * model )
{
2007-05-01 22:32:43 +02:00
strcpy ( model , amiga_model_name ) ;
2005-04-16 15:20:36 -07:00
}
2008-10-03 22:42:36 +04:00
static void amiga_get_hardware_list ( struct seq_file * m )
2005-04-16 15:20:36 -07:00
{
2007-05-01 22:32:43 +02:00
if ( AMIGAHW_PRESENT ( CHIP_RAM ) )
2008-10-03 22:42:36 +04:00
seq_printf ( m , " Chip RAM: \t %ldK \n " , amiga_chip_size > > 10 ) ;
seq_printf ( m , " PS Freq: \t %dHz \n EClock Freq: \t %ldHz \n " ,
2007-05-01 22:32:43 +02:00
amiga_psfreq , amiga_eclock ) ;
if ( AMIGAHW_PRESENT ( AMI_VIDEO ) ) {
char * type ;
switch ( amiga_chipset ) {
case CS_OCS :
type = " OCS " ;
break ;
case CS_ECS :
type = " ECS " ;
break ;
case CS_AGA :
type = " AGA " ;
break ;
default :
type = " Old or Unknown " ;
break ;
}
2008-10-03 22:42:36 +04:00
seq_printf ( m , " Graphics: \t %s \n " , type ) ;
2005-04-16 15:20:36 -07:00
}
# define AMIGAHW_ANNOUNCE(name, str) \
2007-05-01 22:32:43 +02:00
if ( AMIGAHW_PRESENT ( name ) ) \
2008-10-03 22:42:36 +04:00
seq_printf ( m , " \t %s \n " , str )
2007-05-01 22:32:43 +02:00
2016-10-23 22:55:03 +02:00
seq_puts ( m , " Detected hardware: \n " ) ;
2007-05-01 22:32:43 +02:00
AMIGAHW_ANNOUNCE ( AMI_VIDEO , " Amiga Video " ) ;
AMIGAHW_ANNOUNCE ( AMI_BLITTER , " Blitter " ) ;
AMIGAHW_ANNOUNCE ( AMBER_FF , " Amber Flicker Fixer " ) ;
AMIGAHW_ANNOUNCE ( AMI_AUDIO , " Amiga Audio " ) ;
AMIGAHW_ANNOUNCE ( AMI_FLOPPY , " Floppy Controller " ) ;
AMIGAHW_ANNOUNCE ( A3000_SCSI , " SCSI Controller WD33C93 (A3000 style) " ) ;
AMIGAHW_ANNOUNCE ( A4000_SCSI , " SCSI Controller NCR53C710 (A4000T style) " ) ;
AMIGAHW_ANNOUNCE ( A1200_IDE , " IDE Interface (A1200 style) " ) ;
AMIGAHW_ANNOUNCE ( A4000_IDE , " IDE Interface (A4000 style) " ) ;
AMIGAHW_ANNOUNCE ( CD_ROM , " Internal CD ROM drive " ) ;
AMIGAHW_ANNOUNCE ( AMI_KEYBOARD , " Keyboard " ) ;
AMIGAHW_ANNOUNCE ( AMI_MOUSE , " Mouse Port " ) ;
AMIGAHW_ANNOUNCE ( AMI_SERIAL , " Serial Port " ) ;
AMIGAHW_ANNOUNCE ( AMI_PARALLEL , " Parallel Port " ) ;
AMIGAHW_ANNOUNCE ( A2000_CLK , " Hardware Clock (A2000 style) " ) ;
AMIGAHW_ANNOUNCE ( A3000_CLK , " Hardware Clock (A3000 style) " ) ;
AMIGAHW_ANNOUNCE ( CHIP_RAM , " Chip RAM " ) ;
AMIGAHW_ANNOUNCE ( PAULA , " Paula 8364 " ) ;
AMIGAHW_ANNOUNCE ( DENISE , " Denise 8362 " ) ;
AMIGAHW_ANNOUNCE ( DENISE_HR , " Denise 8373 " ) ;
AMIGAHW_ANNOUNCE ( LISA , " Lisa 8375 " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_PAL , " Normal/Fat PAL Agnus 8367/8371 " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_NTSC , " Normal/Fat NTSC Agnus 8361/8370 " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_HR_PAL , " Fat Hires PAL Agnus 8372 " ) ;
AMIGAHW_ANNOUNCE ( AGNUS_HR_NTSC , " Fat Hires NTSC Agnus 8372 " ) ;
AMIGAHW_ANNOUNCE ( ALICE_PAL , " PAL Alice 8374 " ) ;
AMIGAHW_ANNOUNCE ( ALICE_NTSC , " NTSC Alice 8374 " ) ;
AMIGAHW_ANNOUNCE ( MAGIC_REKICK , " Magic Hard Rekick " ) ;
AMIGAHW_ANNOUNCE ( PCMCIA , " PCMCIA Slot " ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_ZORRO
2007-05-01 22:32:43 +02:00
if ( AMIGAHW_PRESENT ( ZORRO ) )
2008-10-03 22:42:36 +04:00
seq_printf ( m , " \t Zorro II%s AutoConfig: %d Expansion "
2007-05-01 22:32:43 +02:00
" Device%s \n " ,
AMIGAHW_PRESENT ( ZORRO3 ) ? " I " : " " ,
zorro_num_autocon , zorro_num_autocon = = 1 ? " " : " s " ) ;
2005-04-16 15:20:36 -07:00
# endif /* CONFIG_ZORRO */
# undef AMIGAHW_ANNOUNCE
}
2008-10-13 21:58:53 +02:00
/*
* The Amiga keyboard driver needs key_maps , but we cannot export it in
* drivers / char / defkeymap . c , as it is autogenerated
*/
# ifdef CONFIG_HW_CONSOLE
EXPORT_SYMBOL_GPL ( key_maps ) ;
# endif