1
0
mirror of https://git.code.sf.net/p/fuse-emulator/fuse synced 2026-01-27 01:41:34 +03:00
Files
fuse/peripherals/multiface.c
Fredrick Meunier 160b7e7544 Always set peripheral status to match snapshots
when they are being loaded
2018-04-03 21:23:16 +10:00

681 lines
18 KiB
C

/* multiface.c: Multiface One/128/3 handling routines
Copyright (c) 2005,2007 Gergely Szasz
Copyright (c) 2017 Fredrick Meunier
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
Author contact information:
Gergely: szaszg@hu.inter.net
Many thanks to: Mark Woodmass
*/
#include <config.h>
#include <string.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <time.h>
#include "debugger/debugger.h"
#include "event.h"
#include "infrastructure/startup_manager.h"
#include "memory.h"
#include "module.h"
#include "multiface.h"
#include "options.h"
#include "periph.h"
#include "settings.h"
#include "ui/ui.h"
#include "unittests/unittests.h"
#include "utils.h"
#include "z80/z80.h"
/* 8KB ROM */
#define MULTIFACE_ROM_SIZE 0x2000
/* 8KB RAM */
#define MULTIFACE_RAM_SIZE 0x2000
#define MF_MASK( a ) ( 1 << a )
#ifdef IS
#undef IS
#endif
#define IS(a, b) (a & MF_MASK( b ) )
#ifdef SET
#undef SET
#endif
#define SET(a, b, c) a = ( a & ~MF_MASK( b ) ) | ( c ? MF_MASK( b ) : 0 )
enum {
MF_1 = 0,
MF_128,
MF_3
};
/* Two 8KB memory chunk accessible by the Z80 when /ROMCS is low */
static memory_page multiface_memory_map_romcs_rom[MEMORY_PAGES_IN_8K];
static memory_page multiface_memory_map_romcs_ram[MEMORY_PAGES_IN_8K];
static int romcs = 0;
typedef struct multiface_t {
int IC8a_Q; /* IC8 74LS74 first Flip-flop /Q output*/
int IC8b_Q; /* IC8 74LS74 second Flip-flop /Q output */
int J2; /* Jumper 2 to disable software paging, or
the software on/off state for 128/3 */
int J1; /* Jumper 1 to disable joystick (always 0) */
libspectrum_byte xfdd_reg[4]; /* 74LS670 chip store low 4 bits of
0x1ffd, 0x3ffd, 0x5ffd and 0x7ffd */
periph_type type; /* type of multiface: one/128/3 */
libspectrum_byte ram[8192]; /* 8k RAM */
int *c_settings; /* ptr to current_settings.multiface### */
char **d_rom;
char **c_rom;
} multiface_t;
static multiface_t mf[3];
int multiface_active = 0;
int multiface_activated = 0;
int multiface_available = 0;
/* Memory source */
static int multiface_rom_memory_source, multiface_ram_memory_source;
/* Debugger events */
static const char * const event_type_string = "multiface";
static int page_event, unpage_event;
static int multiface_init( void *context );
static void multiface_page( int idx );
static void multiface_unpage( int idx );
static void multiface_reset( int hard_reset );
static void multiface_memory_map( void );
static libspectrum_byte multiface_port_in1( libspectrum_word port,
libspectrum_byte *attached );
static libspectrum_byte multiface_port_in128( libspectrum_word port,
libspectrum_byte *attached );
static libspectrum_byte multiface_port_in3( libspectrum_word port,
libspectrum_byte *attached );
static void multiface_port_out1( libspectrum_word port, libspectrum_byte val );
static void multiface_port_out128( libspectrum_word port,
libspectrum_byte val );
static void multiface_port_out3( libspectrum_word port, libspectrum_byte val );
static void multiface_port_xffd_write( libspectrum_word port,
libspectrum_byte val );
static void multiface_enabled_snapshot( libspectrum_snap *snap );
static void multiface_from_snapshot( libspectrum_snap *snap );
static void multiface_to_snapshot( libspectrum_snap *snap );
static module_info_t multiface_module_info = {
/* .reset = */ multiface_reset,
/* .romcs = */ multiface_memory_map,
/* .snapshot_enabled = */ multiface_enabled_snapshot,
/* .snapshot_from = */ multiface_from_snapshot,
/* .snapshot_to = */ multiface_to_snapshot,
};
static const periph_port_t multiface_ports_1[] = {
/* ---- ---- x001 --1- */
{ 0x0072, 0x0012, multiface_port_in1, multiface_port_out1 },
{ 0, 0, NULL, NULL }
};
static const periph_port_t multiface_ports_128[] = {
/* ---- ---- x011 --1- */
{ 0x0072, 0x0032, multiface_port_in128, multiface_port_out128 },
{ 0, 0, NULL, NULL }
};
static const periph_port_t multiface_ports_3[] = {
/* ---- ---- x011 --1- */
{ 0x0072, 0x0032, multiface_port_in3, multiface_port_out3 },
{ 0x90ff, 0x10fd, NULL, multiface_port_xffd_write },
{ 0, 0, NULL, NULL }
};
static const periph_t multiface_periph_1 = {
&settings_current.multiface1,
multiface_ports_1,
1,
NULL
};
static const periph_t multiface_periph_128 = {
&settings_current.multiface128,
multiface_ports_128,
1,
NULL
};
static const periph_t multiface_periph_3 = {
&settings_current.multiface3,
multiface_ports_3,
1,
NULL
};
void
multiface_register_startup( void )
{
startup_manager_module dependencies[] = {
STARTUP_MANAGER_MODULE_DEBUGGER,
STARTUP_MANAGER_MODULE_MEMORY,
STARTUP_MANAGER_MODULE_SETUID,
};
startup_manager_register( STARTUP_MANAGER_MODULE_MULTIFACE, dependencies,
ARRAY_SIZE( dependencies ), multiface_init, NULL,
NULL );
}
static int
multiface_init( void *context GCC_UNUSED )
{
int i;
multiface_active = 0;
multiface_activated = 0;
multiface_available = 0;
module_register( &multiface_module_info );
multiface_rom_memory_source = memory_source_register( "Multiface ROM" );
multiface_ram_memory_source = memory_source_register( "Multiface RAM" );
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ )
multiface_memory_map_romcs_rom[i].source = multiface_rom_memory_source;
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ )
multiface_memory_map_romcs_ram[i].source = multiface_ram_memory_source;
mf[MF_1].type = PERIPH_TYPE_MULTIFACE_1;
mf[MF_128].type = PERIPH_TYPE_MULTIFACE_128;
mf[MF_3].type = PERIPH_TYPE_MULTIFACE_3;
mf[MF_1].c_settings = &settings_current.multiface1;
mf[MF_128].c_settings = &settings_current.multiface128;
mf[MF_3].c_settings = &settings_current.multiface3;
mf[MF_1].d_rom = &settings_default.rom_multiface1;
mf[MF_128].d_rom = &settings_default.rom_multiface128;
mf[MF_3].d_rom = &settings_default.rom_multiface3;
mf[MF_1].c_rom = &settings_current.rom_multiface1;
mf[MF_128].c_rom = &settings_current.rom_multiface128;
mf[MF_3].c_rom = &settings_current.rom_multiface3;
periph_register( PERIPH_TYPE_MULTIFACE_1, &multiface_periph_1 );
periph_register( PERIPH_TYPE_MULTIFACE_128, &multiface_periph_128 );
periph_register( PERIPH_TYPE_MULTIFACE_3, &multiface_periph_3 );
periph_register_paging_events( event_type_string, &page_event,
&unpage_event );
return 0;
}
static void
multiface_reset_real( int idx, int hard_reset )
{
int i;
multiface_unpage( idx );
SET( multiface_activated, idx, 0 );
SET( multiface_available, idx, 0 );
if( hard_reset ) memset( mf[idx].ram, 0, 8192 );
if( !periph_is_active( mf[idx].type ) ) return;
mf[idx].IC8a_Q = 1;
mf[idx].IC8b_Q = 1;
mf[idx].J1 = 0; /* Joystick always disabled :-( */
if( mf[idx].type == PERIPH_TYPE_MULTIFACE_1 )
mf[idx].J2 = settings_current.multiface1_stealth ? 0 : 1;
else
mf[idx].J2 = 0;
memset( mf[idx].xfdd_reg, 0x00, 4 );
*mf[idx].c_settings = 0;
periph_activate_type( mf[idx].type, 0 );
if( machine_load_rom_bank( multiface_memory_map_romcs_rom, 0,
*mf[idx].c_rom, *mf[idx].d_rom, 0x2000 ) )
return;
machine_current->ram.romcs = 0;
/* Now, the last (if all enabled: MF_3) is win (ROM/RAM)
BTW: not too critical, because if only one selected
than have to works this stuff properly...
*/
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ ) {
struct memory_page *page = &multiface_memory_map_romcs_ram[ i ];
page->page = &mf[idx].ram[ i * MEMORY_PAGE_SIZE ];
page->offset = i * MEMORY_PAGE_SIZE;
page->writable = 1;
}
*mf[idx].c_settings = 1;
SET( multiface_available, idx, 1 );
periph_activate_type( mf[idx].type, 1 );
ui_menu_activate( UI_MENU_ITEM_MACHINE_MULTIFACE, 1 );
}
static void
multiface_reset( int hard_reset )
{
multiface_reset_real( MF_1, hard_reset );
multiface_reset_real( MF_128, hard_reset );
multiface_reset_real( MF_3, hard_reset );
ui_menu_activate( UI_MENU_ITEM_MACHINE_MULTIFACE,
( multiface_available ? 1 : 0 ) );
}
void
multiface_status_update( void )
{
int i;
ui_menu_activate( UI_MENU_ITEM_MACHINE_MULTIFACE, 0 );
for( i = MF_1; i <= MF_3; i++ )
SET( multiface_available, i, periph_is_active( mf[i].type ) );
if( !multiface_available ) return;
ui_menu_activate( UI_MENU_ITEM_MACHINE_MULTIFACE, 1 );
if( IS( multiface_available, MF_1 ) &&
mf[MF_1].J2 == settings_current.multiface1_stealth ) {
mf[MF_1].J2 = settings_current.multiface1_stealth ? 0 : 1;
}
}
static void
multiface_page( int idx )
{
if( IS( multiface_active, idx ) ) return;
SET( multiface_active, idx, 1 );
romcs = machine_current->ram.romcs;
machine_current->ram.romcs = 1;
machine_current->memory_map();
debugger_event( page_event );
if( mf[idx].type != PERIPH_TYPE_MULTIFACE_1 )
mf[idx].J2 = 1;
}
static void
multiface_unpage( int idx )
{
if( !IS( multiface_active, idx ) ) return;
SET( multiface_active, idx, 0 );
machine_current->ram.romcs = romcs;
machine_current->memory_map();
debugger_event( unpage_event );
}
static void
multiface_memory_map( void )
{
if( !multiface_active )
return;
memory_map_romcs_8k( 0x0000, multiface_memory_map_romcs_rom );
memory_map_romcs_8k( 0x2000, multiface_memory_map_romcs_ram );
}
static libspectrum_byte
multiface_port_in1( libspectrum_word port, libspectrum_byte *attached )
{
libspectrum_byte ret = 0xff;
int a7;
if( !IS( multiface_available, MF_1 ) ) return ret;
/* TODO: check if this value should be set to 0xff */
*attached = 0xff;
/* in () */
/* Multiface one */
/* xxxxxxxx 1001xx1x page in IN A, (159) 10011111*/
/* xxxxxxxx 0001xx1x page out IN A, (31) 00011111*/
a7 = port & 0x0080;
/* TODO: read joystick */
/*
if( mf[MF_1].J1 ) {
}
*/
if( a7 ) {
if( mf[MF_1].J2 ) {
multiface_page( MF_1 );
mf[MF_1].IC8a_Q = 0;
}
}
else {
multiface_unpage( MF_1 ); /* a7 == 0 */
mf[MF_1].IC8a_Q = 1;
}
return ret;
}
static libspectrum_byte
multiface_port_in128( libspectrum_word port, libspectrum_byte *attached )
{
libspectrum_byte ret = 0xff;
int a7;
if( !IS( multiface_available, MF_128 ) ) return ret;
/* TODO: check if this value should be set to 0xff */
*attached = 0xff;
/* Multiface 128 */
/* I have only the MF128 user guide which say: */
/* IN A, (191) -> page in, and IN A, (63) page out */
/* let see: */
/* xxxxxxxx 10111111 ok, may a0 don't care, so */
/* xxxxxxxx 1011111x and may have some other don't care bit, but how know? */
/**/
/* xxxxxxxx 00111111 ok, may a0 don't care, so */
/* xxxxxxxx 0011111x and may have some other don't care bit, but how know? */
a7 = port & 0x0080;
if( a7 ) {
if( mf[MF_128].J2 ) {
multiface_page( MF_128 );
ret = machine_current->ram.last_byte & 0x08 ? 0xff : 0x7f;
mf[MF_128].IC8a_Q = 0;
}
} else {
multiface_unpage( MF_128 ); /* a7 == 0 */
mf[MF_128].IC8a_Q = 1;
}
return ret;
}
static libspectrum_byte
multiface_port_in3( libspectrum_word port, libspectrum_byte *attached )
{
libspectrum_byte ret = 0xff;
int a7;
if( !IS( multiface_available, MF_3 ) ) return ret;
/* TODO: check if this value should be set to 0xff */
*attached = 0xff;
/* Multiface 3 */
/* The MF3 user guide say nothing about paging memory :-( */
/* The following links say */
/* http://x128.speccy.cz/multiface/multiface.htm */
/* https://www.worldofspectrum.org/forums/discussion/comment/303231/#Comment_303231 */
/* IN A, (63) -> page in, and IN A, (191) page out */
a7 = port & 0x0080;
if( a7 ) { /* a7 == 1 */
multiface_unpage( MF_3 );
mf[MF_3].IC8a_Q = 0;
} else if( mf[MF_3].J2 ) {
multiface_page( MF_3 ); /* a7 == 0 */
mf[MF_3].IC8a_Q = 1;
}
/* Return last data written to port 0x1ffd, 0x3ffd, 0x5ffd or 0x7ffd */
if( mf[MF_3].J2 )
ret = mf[MF_3].xfdd_reg[ ( port & 0x6000 ) >> 13 ] | 0xf0;
return ret;
}
static void
multiface_port_out1( libspectrum_word port GCC_UNUSED,
libspectrum_byte val GCC_UNUSED )
{
if( !IS( multiface_available, MF_1 ) ) return;
/* MF one: out () */
/* xxxxxxxx x001xx1x page out */
mf[MF_1].IC8b_Q = 1;
}
static void
multiface_port_out128_3( int idx, libspectrum_word port )
{
if( !IS( multiface_available, idx ) ) return;
if( IS( multiface_active, idx ) ) {
mf[idx].J2 = port & 0x0080 ? 1 : 0; /* A7 == 1 */
}
mf[idx].IC8b_Q = 1;
}
static void
multiface_port_out128( libspectrum_word port, libspectrum_byte val GCC_UNUSED )
{
multiface_port_out128_3( MF_128, port );
}
static void
multiface_port_out3( libspectrum_word port, libspectrum_byte val GCC_UNUSED )
{
multiface_port_out128_3( MF_3, port );
}
static void
multiface_port_xffd_write( libspectrum_word port, libspectrum_byte val )
{
mf[MF_3].xfdd_reg[ ( port & 0x6000 ) >> 13 ] = val & 0x0f;
}
void
multiface_red_button( void )
{
int i;
/*
One RED button for all ;-)
*/
for( i = MF_3; i >= MF_1; i-- ) {
if( !IS( multiface_available, i ) || mf[i].IC8b_Q == 0 ) continue;
/* Note: AFAIK the Multiface One schematics show that the physical switch
(J2) disables paging in OFF state and has no effect on NMI generation.
But the manual states that the interface is unusable while switched OFF.
Until better understanding, NMI generation is disabled to avoid a freeze
in the spectrum machine.
*/
if( i == MF_1 && !mf[MF_1].J2 ) continue;
mf[i].IC8b_Q = 0;
SET( multiface_activated, i, 1 );
event_add( 0, z80_nmi_event ); /* pull /NMI */
break;
}
}
void
multiface_setic8( void )
{
int i;
/*
activate all at once...
*/
for( i = MF_3; i >= MF_1; i-- ) {
if( !IS( multiface_available, i ) || mf[i].IC8b_Q == 1 ) continue;
mf[i].IC8a_Q = 0;
SET( multiface_activated, i, 0 );
multiface_page( i );
break;
}
}
int
multiface_unittest( void )
{
int r = 0;
multiface_page( MF_1 );
r += unittests_assert_8k_page( 0x0000, multiface_rom_memory_source, 0 );
r += unittests_assert_8k_page( 0x2000, multiface_ram_memory_source, 0 );
r += unittests_assert_16k_ram_page( 0x4000, 5 );
r += unittests_assert_16k_ram_page( 0x8000, 2 );
r += unittests_assert_16k_ram_page( 0xc000, 0 );
multiface_unpage( MF_1 );
r += unittests_paging_test_48( 2 );
return r;
}
static void
multiface_enabled_snapshot( libspectrum_snap *snap )
{
settings_current.multiface1 = 0;
settings_current.multiface128 = 0;
settings_current.multiface3 = 0;
if( !libspectrum_snap_multiface_active( snap ) ) return;
if( libspectrum_snap_multiface_model_one( snap ) )
settings_current.multiface1 = 1;
else if( libspectrum_snap_multiface_model_128( snap ) )
settings_current.multiface128 = 1;
else if( libspectrum_snap_multiface_model_3( snap ) )
settings_current.multiface3 = 1;
}
static void
multiface_from_snapshot( libspectrum_snap *snap )
{
int idx;
if( !libspectrum_snap_multiface_active( snap ) ) return;
if( libspectrum_snap_multiface_model_one( snap ) )
idx = MF_1;
else if( libspectrum_snap_multiface_model_128( snap ) )
idx = MF_128;
else if( libspectrum_snap_multiface_model_3( snap ) )
idx = MF_3;
else
return;
if( !IS( multiface_available, idx ) ) return;
/* Multiface with 16 Kb RAM not supported */
if( libspectrum_snap_multiface_ram_length( snap, 0 ) != MULTIFACE_RAM_SIZE ) {
ui_error( UI_ERROR_ERROR, "Only supported Multiface with 8 Kb RAM" );
return;
}
if( libspectrum_snap_multiface_ram( snap, 0 ) ) {
memcpy( mf[idx].ram,
libspectrum_snap_multiface_ram( snap, 0 ), MULTIFACE_RAM_SIZE );
}
if( libspectrum_snap_multiface_paged( snap ) ) {
multiface_page( idx );
mf[idx].IC8a_Q = ( idx == MF_3 )? 1 : 0;
} else {
multiface_unpage( idx );
}
/* Restore status of software lockout (128/3) or physical switch (One) */
switch( idx ) {
case MF_1:
mf[MF_1].J2 = !libspectrum_snap_multiface_disabled( snap );
settings_current.multiface1_stealth = !mf[MF_1].J2;
break;
case MF_128:
case MF_3:
mf[idx].J2 = !libspectrum_snap_multiface_software_lockout( snap );
break;
}
/* Red button status */
if( libspectrum_snap_multiface_red_button_disabled( snap ) ) {
mf[idx].IC8b_Q = 0;
}
/* Multiface 3 - 4x4 bit register */
if( idx == MF_3 ) {
mf[MF_3].xfdd_reg[ 0 ] =
libspectrum_snap_out_plus3_memoryport( snap ) & 0x0f;
mf[MF_3].xfdd_reg[ 3 ] =
libspectrum_snap_out_128_memoryport( snap ) & 0x0f;
}
}
static void
multiface_to_snapshot( libspectrum_snap *snap )
{
libspectrum_byte *buffer;
int idx, i;
if( periph_is_active( PERIPH_TYPE_MULTIFACE_1 ) ) {
libspectrum_snap_set_multiface_model_one( snap, 1 );
idx = MF_1;
} else if( periph_is_active( PERIPH_TYPE_MULTIFACE_128 ) ) {
libspectrum_snap_set_multiface_model_128( snap, 1 );
idx = MF_128;
} else if( periph_is_active( PERIPH_TYPE_MULTIFACE_3 ) ) {
libspectrum_snap_set_multiface_model_3( snap, 1 );
idx = MF_3;
} else {
return;
}
libspectrum_snap_set_multiface_active( snap, 1 );
libspectrum_snap_set_multiface_paged( snap, IS( multiface_active, idx ) );
/* Store status of software lockout (128/3) or physical switch (One) */
switch( idx ) {
case MF_1:
libspectrum_snap_set_multiface_disabled( snap, !mf[MF_1].J2 );
break;
case MF_128:
case MF_3:
libspectrum_snap_set_multiface_software_lockout( snap, !mf[idx].J2 );
break;
}
/* Store red button status */
if( !mf[idx].IC8b_Q ) {
libspectrum_snap_set_multiface_red_button_disabled( snap, 1 );
}
buffer = libspectrum_new( libspectrum_byte, MULTIFACE_RAM_SIZE );
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ )
memcpy( buffer + i * MEMORY_PAGE_SIZE,
multiface_memory_map_romcs_ram[i].page, MEMORY_PAGE_SIZE );
libspectrum_snap_set_multiface_ram( snap, 0, buffer );
libspectrum_snap_set_multiface_ram_length( snap, 0, MULTIFACE_RAM_SIZE );
}