mirror of
https://git.code.sf.net/p/fuse-emulator/fuse
synced 2026-01-27 01:41:34 +03:00
248 lines
6.4 KiB
C
248 lines
6.4 KiB
C
/* usource.c: Routines for handling the Currah uSource interface
|
|
Copyright (c) 2007-2016 Stuart Brady, Philip Kendall
|
|
Copyright (c) 2016 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:
|
|
|
|
Philip: philip-fuse@shadowmagic.org.uk
|
|
|
|
Stuart: stuart.brady@gmail.com
|
|
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <libspectrum.h>
|
|
|
|
#include "compat.h"
|
|
#include "infrastructure/startup_manager.h"
|
|
#include "machine.h"
|
|
#include "memory_pages.h"
|
|
#include "module.h"
|
|
#include "periph.h"
|
|
#include "settings.h"
|
|
#include "unittests/unittests.h"
|
|
#include "usource.h"
|
|
|
|
/* An 8 KiB memory chunk accessible by the Z80 when /ROMCS is low
|
|
* (mirrored in the second 8 KiB when active) */
|
|
static memory_page usource_memory_map_romcs[ MEMORY_PAGES_IN_8K ];
|
|
static int usource_memory_source;
|
|
|
|
int usource_active = 0;
|
|
int usource_available = 0;
|
|
|
|
static void usource_toggle_write( libspectrum_word port,
|
|
libspectrum_byte val );
|
|
static libspectrum_byte usource_toggle_read( libspectrum_word port,
|
|
libspectrum_byte *attached );
|
|
|
|
static void usource_reset( int hard_reset );
|
|
static void usource_enabled_snapshot( libspectrum_snap *snap );
|
|
static void usource_from_snapshot( libspectrum_snap *snap );
|
|
static void usource_to_snapshot( libspectrum_snap *snap );
|
|
static void usource_memory_map( void );
|
|
|
|
static module_info_t usource_module_info = {
|
|
|
|
/* .reset = */ usource_reset,
|
|
/* .romcs = */ usource_memory_map,
|
|
/* .snapshot_enabled = */ usource_enabled_snapshot,
|
|
/* .snapshot_from = */ usource_from_snapshot,
|
|
/* .snapshot_to = */ usource_to_snapshot,
|
|
|
|
};
|
|
|
|
static const periph_port_t usource_ports[] = {
|
|
{ 0xffff, 0x2bae, usource_toggle_read, usource_toggle_write },
|
|
|
|
{ 0, 0, NULL, NULL }
|
|
};
|
|
|
|
static const periph_t usource_periph = {
|
|
/* .option = */ &settings_current.usource,
|
|
/* .ports = */ usource_ports,
|
|
/* .hard_reset = */ 1,
|
|
/* .activate = */ NULL,
|
|
};
|
|
|
|
static int
|
|
usource_init( void *context )
|
|
{
|
|
int i;
|
|
|
|
module_register( &usource_module_info );
|
|
|
|
usource_memory_source = memory_source_register( "uSource" );
|
|
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ )
|
|
usource_memory_map_romcs[i].source = usource_memory_source;
|
|
|
|
periph_register( PERIPH_TYPE_USOURCE, &usource_periph );
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
usource_end( void )
|
|
{
|
|
usource_available = 0;
|
|
}
|
|
|
|
void
|
|
usource_register_startup( void )
|
|
{
|
|
startup_manager_module dependencies[] = {
|
|
STARTUP_MANAGER_MODULE_MEMORY,
|
|
STARTUP_MANAGER_MODULE_SETUID,
|
|
};
|
|
startup_manager_register( STARTUP_MANAGER_MODULE_USOURCE, dependencies,
|
|
ARRAY_SIZE( dependencies ), usource_init, NULL,
|
|
usource_end );
|
|
}
|
|
|
|
static void
|
|
usource_reset( int hard_reset GCC_UNUSED )
|
|
{
|
|
usource_active = 0;
|
|
usource_available = 0;
|
|
|
|
if( !periph_is_active( PERIPH_TYPE_USOURCE ) )
|
|
return;
|
|
|
|
if( machine_load_rom_bank( usource_memory_map_romcs, 0,
|
|
settings_current.rom_usource,
|
|
settings_default.rom_usource, 0x2000 ) ) {
|
|
settings_current.usource = 0;
|
|
periph_activate_type( PERIPH_TYPE_USOURCE, 0 );
|
|
return;
|
|
}
|
|
|
|
machine_current->ram.romcs = 0;
|
|
|
|
usource_available = 1;
|
|
}
|
|
|
|
void
|
|
usource_toggle( void )
|
|
{
|
|
usource_active = !usource_active;
|
|
machine_current->ram.romcs = usource_active;
|
|
machine_current->memory_map();
|
|
}
|
|
|
|
static void
|
|
usource_memory_map( void )
|
|
{
|
|
if( !usource_active ) return;
|
|
|
|
memory_map_romcs_8k( 0x0000, usource_memory_map_romcs );
|
|
memory_map_romcs_8k( 0x2000, usource_memory_map_romcs );
|
|
}
|
|
|
|
static libspectrum_byte
|
|
usource_toggle_read( libspectrum_word port GCC_UNUSED,
|
|
libspectrum_byte *attached GCC_UNUSED )
|
|
{
|
|
usource_toggle();
|
|
|
|
return 0xff;
|
|
}
|
|
|
|
static void
|
|
usource_toggle_write( libspectrum_word port GCC_UNUSED, libspectrum_byte val )
|
|
{
|
|
usource_toggle();
|
|
}
|
|
|
|
int
|
|
usource_unittest( void )
|
|
{
|
|
int r = 0;
|
|
|
|
usource_active = 1;
|
|
usource_memory_map();
|
|
|
|
r += unittests_assert_8k_page( 0x0000, usource_memory_source, 0 );
|
|
r += unittests_assert_8k_page( 0x2000, usource_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 );
|
|
|
|
usource_active = 0;
|
|
machine_current->memory_map();
|
|
|
|
r += unittests_paging_test_48( 2 );
|
|
|
|
return r;
|
|
}
|
|
|
|
static void
|
|
usource_enabled_snapshot( libspectrum_snap *snap )
|
|
{
|
|
settings_current.usource = libspectrum_snap_usource_active( snap );
|
|
}
|
|
|
|
static void
|
|
usource_from_snapshot( libspectrum_snap *snap )
|
|
{
|
|
if( !libspectrum_snap_usource_active( snap ) ) return;
|
|
|
|
if( libspectrum_snap_usource_custom_rom( snap ) &&
|
|
libspectrum_snap_usource_rom( snap, 0 ) &&
|
|
machine_load_rom_bank_from_buffer(
|
|
usource_memory_map_romcs, 0,
|
|
libspectrum_snap_usource_rom( snap, 0 ),
|
|
libspectrum_snap_usource_rom_length( snap, 0 ),
|
|
1 ) )
|
|
return;
|
|
|
|
if( libspectrum_snap_usource_paged( snap ) ) {
|
|
usource_active = 0; /* Will be toggled to active next */
|
|
usource_toggle();
|
|
}
|
|
}
|
|
|
|
static void
|
|
usource_to_snapshot( libspectrum_snap *snap )
|
|
{
|
|
libspectrum_byte *buffer;
|
|
size_t rom_length;
|
|
int i;
|
|
|
|
if( !periph_is_active( PERIPH_TYPE_USOURCE ) ) return;
|
|
|
|
libspectrum_snap_set_usource_active( snap, 1 );
|
|
libspectrum_snap_set_usource_paged ( snap, usource_active );
|
|
|
|
if( usource_memory_map_romcs[0].save_to_snapshot ) {
|
|
rom_length = 0x2000;
|
|
|
|
libspectrum_snap_set_usource_custom_rom( snap, 1 );
|
|
libspectrum_snap_set_usource_rom_length( snap, 0, rom_length );
|
|
|
|
buffer = libspectrum_new( libspectrum_byte, rom_length );
|
|
|
|
for( i = 0; i < MEMORY_PAGES_IN_8K; i++ )
|
|
memcpy( buffer + i * MEMORY_PAGE_SIZE,
|
|
usource_memory_map_romcs[ i ].page, MEMORY_PAGE_SIZE );
|
|
|
|
libspectrum_snap_set_usource_rom( snap, 0, buffer );
|
|
}
|
|
}
|