8cb555460f
'swcodec' is now always set (and recording_swcodec for recording-capable units) in feature.txt so the manual and language strings don't need to all be fixed up. Change-Id: Ib2c9d5d157af8d33653e2d4b4a12881b9aa6ddb0
530 lines
14 KiB
C
530 lines
14 KiB
C
/*
|
|
* Copyright (C) 1996-1998 Szeredi Miklos
|
|
* Email: mszeredi@inf.bme.hu
|
|
*
|
|
* 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. See the file COPYING.
|
|
*
|
|
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*
|
|
*/
|
|
#include "zxmisc.h"
|
|
#include "zxconfig.h"
|
|
#include "lib/configfile.h"
|
|
|
|
#include "spperif.h"
|
|
#include "z80.h"
|
|
#include "spmain.h"
|
|
#include "sptiming.h"
|
|
#include "spscr.h"
|
|
#include "spkey_p.h"
|
|
#include "spkey.h"
|
|
#include "sptape.h"
|
|
#include "spsound.h"
|
|
#include "snapshot.h"
|
|
|
|
#include "spconf.h"
|
|
|
|
#include "interf.h"
|
|
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#ifdef USE_GREY
|
|
#include "lib/grey.h"
|
|
#endif
|
|
|
|
#include "zxbox_keyb.h"
|
|
|
|
int endofsingle IBSS_ATTR;
|
|
|
|
int sp_nosync IBSS_ATTR = 0;
|
|
|
|
int showframe IBSS_ATTR = 1;
|
|
int load_immed = 1;
|
|
|
|
qbyte sp_int_ctr IBSS_ATTR = 0;
|
|
int intkeys[5] IBSS_ATTR;
|
|
|
|
#ifdef USE_DJGPP
|
|
#define DOS
|
|
#endif
|
|
|
|
#define GLOBALCFG "zxbox.cfg"
|
|
|
|
/* here goes settings hadling */
|
|
|
|
/*static struct zxbox_settings settings;*/
|
|
static struct zxbox_settings old_settings;
|
|
|
|
static char* noyes_options[] = { "No", "Yes" };
|
|
/*static char* kempston_options[] = { "No", "Yes" };
|
|
static char* showfps_options[] = {"No", "Yes"};*/
|
|
|
|
static struct configdata config[] =
|
|
{
|
|
{TYPE_ENUM, 0, 2, { .int_p = &settings.invert_colors }, "Invert Colors",
|
|
noyes_options},
|
|
{TYPE_ENUM, 0, 2, { .int_p = &settings.kempston }, "Map keys to kempston",
|
|
noyes_options},
|
|
{TYPE_ENUM, 0, 2, { .int_p = &settings.showfps }, "Show Speed",
|
|
noyes_options},
|
|
{TYPE_ENUM, 0, 2, { .int_p = &settings.sound }, "Sound", noyes_options},
|
|
{TYPE_INT, 0, 9, { .int_p = &settings.frameskip }, "Frameskip", NULL},
|
|
{TYPE_INT, 1, 10, { .int_p = &settings.volume }, "Volume", NULL},
|
|
{TYPE_STRING, 0, 5, { .string = (char*)&settings.keymap }, "Key Mapping",
|
|
NULL},
|
|
};
|
|
int spcf_read_conf_file(const char *filename)
|
|
{
|
|
settings.volume = 10;
|
|
settings.showfps=1;
|
|
settings.keymap[0]='2';
|
|
settings.keymap[1]='w';
|
|
settings.keymap[2]='9';
|
|
settings.keymap[3]='0';
|
|
settings.keymap[4]='z';
|
|
settings.kempston = 1 ;
|
|
settings.invert_colors=0;
|
|
settings.sound = 0;
|
|
settings.frameskip = 0;
|
|
|
|
|
|
if (configfile_load(filename, config,
|
|
sizeof(config)/sizeof(*config),
|
|
SETTINGS_MIN_VERSION
|
|
) < 0)
|
|
{
|
|
/* If the loading failed, save a new config file (as the disk is
|
|
already spinning) */
|
|
configfile_save(filename, config,
|
|
sizeof(config)/sizeof(*config),
|
|
SETTINGS_VERSION);
|
|
}
|
|
/* Keep a copy of the saved version of the settings - so we can check if
|
|
the settings have changed when we quit */
|
|
old_settings = settings;
|
|
sound_on = settings.sound;
|
|
showframe = settings.frameskip+1;
|
|
int i;
|
|
for ( i=0 ; i<5 ; i++){
|
|
if (settings.keymap[i] == 'E')
|
|
intkeys[i]=SK_KP_Enter;
|
|
else if ( settings.keymap[i] == 'S' )
|
|
intkeys[i]=SK_KP_Space;
|
|
else
|
|
intkeys[i] = (unsigned) settings.keymap[i];
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/* and here it stops (settings loading ;) )*/
|
|
|
|
/* set keys */
|
|
static void set_keys(void){
|
|
char c;
|
|
int selected=0;
|
|
int result;
|
|
int menu_quit=0;
|
|
MENUITEM_STRINGLIST(menu, "Custom keymap", NULL,
|
|
"Map Up key", "Map Down key", "Map Left key",
|
|
"Map Right key", "Map Fire/Jump key");
|
|
|
|
rb->button_clear_queue();
|
|
|
|
while (!menu_quit) {
|
|
result = rb->do_menu(&menu, &selected, NULL, false);
|
|
|
|
switch(result)
|
|
{
|
|
case 0:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
settings.keymap[0]=c;
|
|
}
|
|
break;
|
|
case 1:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
settings.keymap[1]=c;
|
|
}
|
|
break;
|
|
case 2:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
settings.keymap[2]=c;
|
|
}
|
|
break;
|
|
case 3:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
settings.keymap[3]=c;
|
|
}
|
|
break;
|
|
case 4:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
settings.keymap[4]=c;
|
|
}
|
|
break;
|
|
default:
|
|
menu_quit=1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* select predefined keymap */
|
|
static void select_keymap(void){
|
|
int selected=0;
|
|
int result;
|
|
int menu_quit=0;
|
|
MENUITEM_STRINGLIST(menu, "Predefined keymap", NULL,
|
|
"2w90z", "qaopS", "7658S");
|
|
|
|
rb->button_clear_queue();
|
|
|
|
while (!menu_quit) {
|
|
result = rb->do_menu(&menu, &selected, NULL, false);
|
|
|
|
switch(result)
|
|
{
|
|
case 0:
|
|
rb->memcpy ( (void*)&settings.keymap[0] , (void*)"2w90z" , 5);
|
|
menu_quit=1;
|
|
break;
|
|
case 1:
|
|
rb->memcpy ( (void*)&settings.keymap[0] , (void*)"qaopS" , 5);
|
|
menu_quit=1;
|
|
break;
|
|
case 2:
|
|
rb->memcpy ( (void*)&settings.keymap[0] , (void*)"7658S" , 5);
|
|
menu_quit=1;
|
|
break;
|
|
default:
|
|
menu_quit=1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* options menu */
|
|
static void options_menu(void){
|
|
static const struct opt_items no_yes[2] = {
|
|
{ "No", -1 },
|
|
{ "Yes", -1 },
|
|
};
|
|
int selected;
|
|
int result;
|
|
int menu_quit=0;
|
|
int new_setting;
|
|
MENUITEM_STRINGLIST(menu, "Options", NULL,
|
|
"Map Keys to kempston", "Display Speed",
|
|
"Invert Colors", "Frameskip", "Sound", "Volume",
|
|
"Predefined keymap", "Custom keymap");
|
|
|
|
static struct opt_items frameskip_items[] = {
|
|
{ "0", -1 },
|
|
{ "1", -1 },
|
|
{ "2", -1 },
|
|
{ "3", -1 },
|
|
{ "4", -1 },
|
|
{ "5", -1 },
|
|
{ "6", -1 },
|
|
{ "7", -1 },
|
|
{ "8", -1 },
|
|
{ "9", -1 },
|
|
};
|
|
|
|
|
|
rb->button_clear_queue();
|
|
|
|
while (!menu_quit) {
|
|
result = rb->do_menu(&menu, &selected, NULL, false);
|
|
|
|
switch(result)
|
|
{
|
|
case 0:
|
|
new_setting=settings.kempston;
|
|
rb->set_option("Map Keys to kempston",&new_setting,INT,
|
|
no_yes, 2, NULL);
|
|
if (new_setting != settings.kempston )
|
|
settings.kempston=new_setting;
|
|
break;
|
|
case 1:
|
|
new_setting = settings.showfps;
|
|
rb->set_option("Display Speed",&new_setting,INT,
|
|
no_yes, 2, NULL);
|
|
if (new_setting != settings.showfps )
|
|
settings.showfps=new_setting;
|
|
break;
|
|
case 2:
|
|
new_setting = settings.invert_colors;
|
|
rb->set_option("Invert Colors",&new_setting,INT,
|
|
no_yes, 2, NULL);
|
|
if (new_setting != settings.invert_colors )
|
|
settings.invert_colors=new_setting;
|
|
rb->splash(HZ, "Restart to see effect");
|
|
break;
|
|
case 3:
|
|
new_setting = settings.frameskip;
|
|
rb->set_option("Frameskip",&new_setting,INT,
|
|
frameskip_items, 10, NULL);
|
|
if (new_setting != settings.frameskip )
|
|
settings.frameskip=new_setting;
|
|
break;
|
|
case 4:
|
|
new_setting = settings.sound;
|
|
rb->set_option("Sound",&new_setting,INT,
|
|
no_yes, 2, NULL);
|
|
if (new_setting != settings.sound )
|
|
settings.sound=new_setting;
|
|
#if !defined SIMULATOR
|
|
rb->pcm_play_stop();
|
|
#endif
|
|
break;
|
|
case 5:
|
|
new_setting = 9 - settings.volume;
|
|
rb->set_option("Volume",&new_setting,INT,
|
|
frameskip_items, 10, NULL);
|
|
new_setting = 9 - new_setting;
|
|
if (new_setting != settings.volume )
|
|
settings.volume=new_setting;
|
|
break;
|
|
case 6:
|
|
select_keymap();
|
|
break;
|
|
case 7:
|
|
set_keys();
|
|
break;
|
|
default:
|
|
menu_quit=1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* menu */
|
|
static bool zxbox_menu(void)
|
|
{
|
|
#if !defined SIMULATOR
|
|
rb->pcm_play_stop();
|
|
#endif
|
|
int selected=0;
|
|
int result;
|
|
int menu_quit=0;
|
|
int exit=0;
|
|
char c;
|
|
MENUITEM_STRINGLIST(menu, "ZXBox Menu", NULL,
|
|
"VKeyboard", "Play/Pause Tape",
|
|
"Save quick snapshot", "Load quick snapshot",
|
|
"Save Snapshot", "Toggle \"fast\" mode",
|
|
"Options", "Quit");
|
|
|
|
rb->button_clear_queue();
|
|
|
|
while (!menu_quit) {
|
|
result = rb->do_menu(&menu, &selected, NULL, false);
|
|
|
|
switch(result)
|
|
{
|
|
case 0:
|
|
if (!zx_kbd_input((char*) &c))
|
|
{
|
|
press_key(c);
|
|
}
|
|
clear_kbd=1;
|
|
menu_quit=1;
|
|
break;
|
|
case 1:
|
|
pause_play();
|
|
menu_quit=1;
|
|
break;
|
|
case 2:
|
|
save_quick_snapshot();
|
|
menu_quit = 1;
|
|
break;
|
|
case 3:
|
|
load_quick_snapshot();
|
|
menu_quit = 1;
|
|
break;
|
|
case 4:
|
|
save_snapshot();
|
|
break;
|
|
case 5:
|
|
sp_nosync=!sp_nosync;
|
|
menu_quit=1;
|
|
break;
|
|
case 6:
|
|
options_menu();
|
|
break;
|
|
case 7:
|
|
menu_quit=1;
|
|
exit=1;
|
|
break;
|
|
default:
|
|
menu_quit=1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
#if defined(HAVE_ADJUSTABLE_CPU_FREQ)
|
|
rb->cpu_boost(true);
|
|
#endif
|
|
|
|
int i;
|
|
for ( i=0 ; i<5 ; i++){
|
|
if (settings.keymap[i] == 'E')
|
|
intkeys[i]=SK_KP_Enter;
|
|
else if ( settings.keymap[i] == 'S' )
|
|
intkeys[i]=SK_KP_Space;
|
|
else
|
|
intkeys[i] = (unsigned) settings.keymap[i];
|
|
}
|
|
#ifdef USE_GREY
|
|
grey_show(true);
|
|
#endif
|
|
return (exit);
|
|
}
|
|
|
|
|
|
/* */
|
|
extern unsigned char loadim[];
|
|
extern unsigned long loadim_size;
|
|
|
|
#define SHOW_OFFS 1
|
|
|
|
static void update(void)
|
|
{
|
|
update_screen();
|
|
sp_border_update >>= 1;
|
|
sp_imag_vert = sp_imag_horiz = 0;
|
|
}
|
|
|
|
static void run_singlemode(void)
|
|
{
|
|
int t = 0;
|
|
int evenframe, halfsec, updateframe;
|
|
|
|
sp_int_ctr = 0;
|
|
endofsingle = 0;
|
|
spti_reset();
|
|
|
|
while(!endofsingle) {
|
|
video_frames++;
|
|
if (clear_kbd){
|
|
clear_keystates();
|
|
clear_kbd=0;
|
|
}
|
|
if(exit_requested){
|
|
if (zxbox_menu()){
|
|
/* Save the user settings if they have changed */
|
|
if (rb->memcmp(&settings,&old_settings,sizeof(settings))!=0) {
|
|
#ifdef USE_GREY
|
|
grey_show(false);
|
|
#endif
|
|
rb->splash(0, "Saving settings...");
|
|
configfile_save(GLOBALCFG, config,sizeof(config)/sizeof(*config),SETTINGS_VERSION);
|
|
}
|
|
|
|
return;
|
|
}
|
|
exit_requested = 0;
|
|
video_frames=-1;
|
|
start_time = *rb->current_tick;
|
|
sound_on = settings.sound;
|
|
showframe = settings.frameskip+1;
|
|
spti_reset();
|
|
}
|
|
halfsec = !(sp_int_ctr % 25);
|
|
evenframe = !(sp_int_ctr & 1);
|
|
|
|
if(screen_visible) updateframe = sp_nosync ? halfsec :
|
|
!((sp_int_ctr+SHOW_OFFS) % showframe);
|
|
else updateframe = 0;
|
|
if(halfsec) {
|
|
sp_flash_state = ~sp_flash_state;
|
|
flash_change();
|
|
}
|
|
if(evenframe) {
|
|
play_tape();
|
|
sp_scline = 0;
|
|
}
|
|
spkb_process_events(evenframe);
|
|
sp_updating = updateframe;
|
|
t += CHKTICK;
|
|
t = sp_halfframe(t, evenframe ? EVENHF : ODDHF);
|
|
if(SPNM(load_trapped)) {
|
|
SPNM(load_trapped) = 0;
|
|
DANM(haltstate) = 0;
|
|
qload();
|
|
}
|
|
z80_interrupt(0xFF);
|
|
sp_int_ctr++;
|
|
if(!evenframe) rec_tape();
|
|
if(!sp_nosync) {
|
|
if(!sound_avail) spti_wait();
|
|
|
|
if(updateframe) update();
|
|
play_sound(evenframe);
|
|
}
|
|
else if(updateframe) update();
|
|
}
|
|
|
|
}
|
|
|
|
|
|
void check_params(const void* parameter)
|
|
{
|
|
spcf_read_conf_file(GLOBALCFG);
|
|
spcf_read_command_line(parameter);
|
|
}
|
|
|
|
static void init_load(const void *parameter)
|
|
{
|
|
if(load_immed) snsh_z80_load_intern(loadim, loadim_size);
|
|
|
|
check_params(parameter);
|
|
if(spcf_init_snapshot != NULL) {
|
|
#ifndef USE_GREY
|
|
rb->splashf(HZ, "Loading snapshot '%s'", spcf_init_snapshot);
|
|
#endif
|
|
|
|
load_snapshot_file_type(spcf_init_snapshot, spcf_init_snapshot_type);
|
|
free_string(spcf_init_snapshot);
|
|
}
|
|
|
|
if(spcf_init_tapefile != NULL) {
|
|
/*sprintf(msgbuf, "Loading tape '%s'", spcf_init_tapefile);
|
|
put_msg(msgbuf);*/
|
|
start_play_file_type(spcf_init_tapefile, 0, spcf_init_tapefile_type);
|
|
if(!load_immed) pause_play();
|
|
free_string(spcf_init_tapefile);
|
|
}
|
|
}
|
|
|
|
void start_spectemu(const void *parameter)
|
|
{
|
|
spti_init();
|
|
init_load(parameter);
|
|
init_spect_scr();
|
|
init_spect_sound();
|
|
init_spect_key();
|
|
|
|
run_singlemode();
|
|
}
|
|
|
|
|
|
|
|
|