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/if2.c
Fredrick Meunier ba77c2ab67 Move memory.[ch] to memory_pages.[ch]
To avoid clashes with any similarly named system files (thanks,
BogDan Vatra and Sergio Baldoví).
2016-12-10 20:27:03 +11:00

227 lines
5.4 KiB
C

/* if2.c: Interface 2 cartridge handling routines
Copyright (c) 2003-2016 Darren Salt, Fredrick Meunier, Philip Kendall
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:
E-mail: philip-fuse@shadowmagic.org.uk
Darren: linux@youmustbejoking.demon.co.uk
Fred: fredm@spamcop.net
*/
#include <config.h>
#include <string.h>
#include "if2.h"
#include "infrastructure/startup_manager.h"
#include "machine.h"
#include "memory_pages.h"
#include "module.h"
#include "periph.h"
#include "settings.h"
#include "ui/ui.h"
#include "unittests/unittests.h"
/* A 16KB memory chunk accessible by the Z80 when /ROMCS is low */
static memory_page if2_memory_map_romcs[MEMORY_PAGES_IN_16K];
/* IF2 cart inserted? */
int if2_active = 0;
/* IF2 memory source */
static int if2_memory_source;
static void if2_reset( int hard_reset );
static void if2_memory_map( void );
static void if2_from_snapshot( libspectrum_snap *snap );
static void if2_to_snapshot( libspectrum_snap *snap );
static module_info_t if2_module_info = {
/* .reset = */ if2_reset,
/* .romcs = */ if2_memory_map,
/* .snapshot_enabled = */ NULL,
/* .snapshot_from = */ if2_from_snapshot,
/* .snapshot_to = */ if2_to_snapshot,
};
static const periph_t if2_periph = {
/* .option = */ &settings_current.interface2,
/* .ports = */ NULL,
/* .hard_reset = */ 0,
/* .activate = */ NULL,
};
static int
if2_init( void *context )
{
int i;
int if2_source;
module_register( &if2_module_info );
if2_source = memory_source_register( "If2" );
for( i = 0; i < MEMORY_PAGES_IN_16K; i++ )
if2_memory_map_romcs[i].source = if2_source;
periph_register( PERIPH_TYPE_INTERFACE2, &if2_periph );
return 0;
}
void
if2_register_startup( void )
{
startup_manager_module dependencies[] = {
STARTUP_MANAGER_MODULE_MEMORY,
STARTUP_MANAGER_MODULE_SETUID,
};
startup_manager_register( STARTUP_MANAGER_MODULE_IF2, dependencies,
ARRAY_SIZE( dependencies ), if2_init, NULL, NULL );
}
int
if2_insert( const char *filename )
{
if ( !periph_is_active( PERIPH_TYPE_INTERFACE2 ) ) {
ui_error( UI_ERROR_ERROR,
"This machine does not support the Interface 2" );
return 1;
}
settings_set_string( &settings_current.if2_file, filename );
machine_reset( 0 );
return 0;
}
void
if2_eject( void )
{
if ( !periph_is_active( PERIPH_TYPE_INTERFACE2 ) ) {
ui_error( UI_ERROR_ERROR,
"This machine does not support the Interface 2" );
return;
}
if( settings_current.if2_file ) libspectrum_free( settings_current.if2_file );
settings_current.if2_file = NULL;
machine_current->ram.romcs = 0;
ui_menu_activate( UI_MENU_ITEM_MEDIA_CARTRIDGE_IF2_EJECT, 0 );
machine_reset( 0 );
}
static void
if2_reset( int hard_reset GCC_UNUSED )
{
if2_active = 0;
if( !settings_current.if2_file ) {
ui_menu_activate( UI_MENU_ITEM_MEDIA_CARTRIDGE_IF2_EJECT, 0 );
return;
}
if ( !periph_is_active( PERIPH_TYPE_INTERFACE2 ) ) return;
if ( machine_load_rom_bank( if2_memory_map_romcs, 0,
settings_current.if2_file,
NULL, 0x4000 ) )
return;
machine_current->ram.romcs = 1;
if2_active = 1;
memory_romcs_map();
ui_menu_activate( UI_MENU_ITEM_MEDIA_CARTRIDGE_IF2_EJECT, 1 );
}
static void
if2_memory_map( void )
{
if( !if2_active ) return;
memory_map_romcs_full( if2_memory_map_romcs );
}
static void
if2_from_snapshot( libspectrum_snap *snap )
{
if( !libspectrum_snap_interface2_active( snap ) ) return;
if2_active = 1;
machine_current->ram.romcs = 1;
if( libspectrum_snap_interface2_rom( snap, 0 ) &&
machine_load_rom_bank_from_buffer(
if2_memory_map_romcs, 0,
libspectrum_snap_interface2_rom( snap, 0 ),
0x4000,
1 ) )
return;
ui_menu_activate( UI_MENU_ITEM_MEDIA_CARTRIDGE_IF2_EJECT, 1 );
machine_current->memory_map();
}
static void
if2_to_snapshot( libspectrum_snap *snap )
{
libspectrum_byte *buffer;
int i;
if( !if2_active ) return;
libspectrum_snap_set_interface2_active( snap, 1 );
buffer = libspectrum_new( libspectrum_byte, 0x4000 );
for( i = 0; i < MEMORY_PAGES_IN_16K; i++ )
memcpy( buffer + i * MEMORY_PAGE_SIZE,
if2_memory_map_romcs[ i ].page, MEMORY_PAGE_SIZE );
libspectrum_snap_set_interface2_rom( snap, 0, buffer );
}
int
if2_unittest( void )
{
int r = 0;
if2_active = 1;
machine_current->memory_map();
r += unittests_assert_16k_page( 0x0000, if2_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 );
if2_active = 0;
machine_current->memory_map();
r += unittests_paging_test_48( 2 );
return r;
}