648 lines
15 KiB
C
Raw Normal View History

2008-08-07 12:43:17 +00:00
#include <windows.h>
2009-11-22 16:35:41 +00:00
#include "SDL.h"
#include "SDL_thread.h"
2008-08-26 15:58:27 +00:00
2008-08-07 12:43:17 +00:00
#include "shared.h"
2008-10-13 16:07:02 +00:00
#include "sms_ntsc.h"
#include "md_ntsc.h"
2008-08-07 12:43:17 +00:00
#define SOUND_FREQUENCY 48000
2009-11-22 16:35:41 +00:00
#define SOUND_SAMPLES_SIZE 2048
2008-08-24 19:58:37 +00:00
2010-12-04 17:16:55 +00:00
#define VIDEO_WIDTH 320
#define VIDEO_HEIGHT 240
2008-08-07 12:43:17 +00:00
2009-11-22 16:35:41 +00:00
int joynum = 0;
2008-08-24 19:58:37 +00:00
2009-11-22 16:35:41 +00:00
int log_error = 0;
int debug_on = 0;
int turbo_mode = 0;
int use_sound = 1;
int fullscreen = 0; /* SDL_FULLSCREEN */
2008-08-24 19:58:37 +00:00
/* sound */
2008-08-24 19:58:37 +00:00
struct {
char* current_pos;
char* buffer;
int current_emulated_samples;
} sdl_sound;
2008-08-07 12:43:17 +00:00
2008-08-24 19:58:37 +00:00
static void sdl_sound_callback(void *userdata, Uint8 *stream, int len)
{
if(sdl_sound.current_emulated_samples < len) {
memset(stream, 0, len);
}
else {
memcpy(stream, sdl_sound.buffer, len);
/* loop to compensate desync */
do {
sdl_sound.current_emulated_samples -= len;
} while(sdl_sound.current_emulated_samples > 2 * len);
memcpy(sdl_sound.buffer,
sdl_sound.current_pos - sdl_sound.current_emulated_samples,
sdl_sound.current_emulated_samples);
sdl_sound.current_pos = sdl_sound.buffer + sdl_sound.current_emulated_samples;
2008-08-26 15:58:27 +00:00
}
2008-08-24 19:58:37 +00:00
}
2008-08-26 15:58:27 +00:00
static int sdl_sound_init()
2008-08-24 19:58:37 +00:00
{
int n;
SDL_AudioSpec as_desired, as_obtained;
2008-08-24 19:58:37 +00:00
if(SDL_Init(SDL_INIT_AUDIO) < 0) {
2009-11-22 16:35:41 +00:00
MessageBox(NULL, "SDL Audio initialization failed", "Error", 0);
2008-08-24 19:58:37 +00:00
return 0;
}
2008-08-25 20:58:05 +00:00
2009-11-22 16:35:41 +00:00
as_desired.freq = SOUND_FREQUENCY;
as_desired.format = AUDIO_S16LSB;
as_desired.channels = 2;
2009-11-22 16:35:41 +00:00
as_desired.samples = SOUND_SAMPLES_SIZE;
as_desired.callback = sdl_sound_callback;
2008-08-24 19:58:37 +00:00
if(SDL_OpenAudio(&as_desired, &as_obtained) == -1) {
2009-11-22 16:35:41 +00:00
MessageBox(NULL, "SDL Audio open failed", "Error", 0);
return 0;
}
if(as_desired.samples != as_obtained.samples) {
2009-11-22 16:35:41 +00:00
MessageBox(NULL, "SDL Audio wrong setup", "Error", 0);
2008-08-26 15:58:27 +00:00
return 0;
2008-08-24 19:58:37 +00:00
}
2008-08-26 15:58:27 +00:00
sdl_sound.current_emulated_samples = 0;
n = SOUND_SAMPLES_SIZE * 2 * sizeof(short) * 11;
sdl_sound.buffer = (char*)malloc(n);
if(!sdl_sound.buffer) {
2009-11-22 16:35:41 +00:00
MessageBox(NULL, "Can't allocate audio buffer", "Error", 0);
return 0;
}
memset(sdl_sound.buffer, 0, n);
sdl_sound.current_pos = sdl_sound.buffer;
2008-08-26 15:58:27 +00:00
return 1;
2008-08-24 19:58:37 +00:00
}
2008-08-07 12:43:17 +00:00
static void sdl_sound_update()
{
int i;
short* p;
2010-01-27 07:12:26 +00:00
int size = audio_update();
if (use_sound)
{
SDL_LockAudio();
p = (short*)sdl_sound.current_pos;
for(i = 0; i < size; ++i) {
*p = snd.buffer[0][i];
++p;
*p = snd.buffer[1][i];
++p;
}
sdl_sound.current_pos = (char*)p;
sdl_sound.current_emulated_samples += size * 2 * sizeof(short);
SDL_UnlockAudio();
}
}
static void sdl_sound_close()
{
SDL_PauseAudio(1);
SDL_CloseAudio();
2009-11-22 16:35:41 +00:00
if (sdl_sound.buffer)
free(sdl_sound.buffer);
}
2009-11-22 16:35:41 +00:00
/* video */
md_ntsc_t *md_ntsc;
sms_ntsc_t *sms_ntsc;
2009-11-22 16:35:41 +00:00
struct {
SDL_Surface* surf_screen;
SDL_Surface* surf_bitmap;
2010-06-30 08:16:20 +00:00
SDL_Rect srect;
SDL_Rect drect;
2009-11-22 16:35:41 +00:00
Uint32 frames_rendered;
} sdl_video;
static int sdl_video_init()
{
if(SDL_InitSubSystem(SDL_INIT_VIDEO) < 0) {
MessageBox(NULL, "SDL Video initialization failed", "Error", 0);
return 0;
}
sdl_video.surf_screen = SDL_SetVideoMode(VIDEO_WIDTH, VIDEO_HEIGHT, 16, SDL_SWSURFACE | fullscreen);
sdl_video.surf_bitmap = SDL_CreateRGBSurface(SDL_SWSURFACE, 720, 576, 16, 0, 0, 0, 0);
sdl_video.frames_rendered = 0;
SDL_ShowCursor(0);
return 1;
}
2008-10-13 16:07:02 +00:00
2009-11-22 16:35:41 +00:00
static void sdl_video_update()
2008-10-13 16:07:02 +00:00
{
2009-11-22 16:35:41 +00:00
system_frame(0);
/* viewport size changed */
2010-06-30 08:16:20 +00:00
if(bitmap.viewport.changed & 1)
{
2010-06-30 08:16:20 +00:00
bitmap.viewport.changed &= ~1;
/* source bitmap */
sdl_video.srect.w = bitmap.viewport.w+2*bitmap.viewport.x;
sdl_video.srect.h = bitmap.viewport.h+2*bitmap.viewport.y;
sdl_video.srect.x = 0;
sdl_video.srect.y = 0;
if (sdl_video.srect.w > VIDEO_WIDTH)
{
sdl_video.srect.x = (sdl_video.srect.w - VIDEO_WIDTH) / 2;
sdl_video.srect.w = VIDEO_WIDTH;
}
if (sdl_video.srect.h > VIDEO_HEIGHT)
2010-12-04 17:16:55 +00:00
{
2010-06-30 08:16:20 +00:00
sdl_video.srect.y = (sdl_video.srect.h - VIDEO_HEIGHT) / 2;
sdl_video.srect.h = VIDEO_HEIGHT;
}
/* destination bitmap */
sdl_video.drect.w = sdl_video.srect.w;
sdl_video.drect.h = sdl_video.srect.h;
sdl_video.drect.x = (VIDEO_WIDTH - sdl_video.drect.w) / 2;
sdl_video.drect.y = (VIDEO_HEIGHT - sdl_video.drect.h) / 2;
2010-12-04 17:16:55 +00:00
2010-06-30 08:16:20 +00:00
/* clear destination surface */
SDL_FillRect(sdl_video.surf_screen, 0, 0);
/*if (config.render && (interlaced || config.ntsc)) rect.h *= 2;
if (config.ntsc) rect.w = (reg[12]&1) ? MD_NTSC_OUT_WIDTH(rect.w) : SMS_NTSC_OUT_WIDTH(rect.w);
if (config.ntsc)
2009-11-22 16:35:41 +00:00
{
sms_ntsc = (sms_ntsc_t *)malloc(sizeof(sms_ntsc_t));
md_ntsc = (md_ntsc_t *)malloc(sizeof(md_ntsc_t));
switch (config.ntsc)
{
case 1:
sms_ntsc_init(sms_ntsc, &sms_ntsc_composite);
md_ntsc_init(md_ntsc, &md_ntsc_composite);
break;
case 2:
sms_ntsc_init(sms_ntsc, &sms_ntsc_svideo);
md_ntsc_init(md_ntsc, &md_ntsc_svideo);
break;
case 3:
sms_ntsc_init(sms_ntsc, &sms_ntsc_rgb);
md_ntsc_init(md_ntsc, &md_ntsc_rgb);
break;
}
2009-11-22 16:35:41 +00:00
}
else
2009-11-22 16:35:41 +00:00
{
if (sms_ntsc)
{
free(sms_ntsc);
sms_ntsc = NULL;
}
if (md_ntsc)
{
free(md_ntsc);
md_ntsc = NULL;
}
2010-06-30 08:16:20 +00:00
} */
}
2009-11-22 16:35:41 +00:00
2010-06-30 08:16:20 +00:00
SDL_BlitSurface(sdl_video.surf_bitmap, &sdl_video.srect, sdl_video.surf_screen, &sdl_video.drect);
2009-11-22 16:35:41 +00:00
SDL_UpdateRect(sdl_video.surf_screen, 0, 0, 0, 0);
++sdl_video.frames_rendered;
2008-10-13 16:07:02 +00:00
}
2009-11-22 16:35:41 +00:00
static void sdl_video_close()
2008-10-13 16:07:02 +00:00
{
if (sdl_video.surf_bitmap)
SDL_FreeSurface(sdl_video.surf_bitmap);
if (sdl_video.surf_screen)
SDL_FreeSurface(sdl_video.surf_screen);
2009-11-22 16:35:41 +00:00
}
/* Timer Sync */
struct {
SDL_sem* sem_sync;
2010-01-27 07:12:26 +00:00
unsigned ticks;
2009-11-22 16:35:41 +00:00
} sdl_sync;
/* sync */
static Uint32 sdl_sync_timer_callback(Uint32 interval)
{
2010-01-27 07:12:26 +00:00
SDL_SemPost(sdl_sync.sem_sync);
char caption[100];
2010-01-27 07:12:26 +00:00
sdl_sync.ticks++;
if (sdl_sync.ticks == (vdp_pal ? 50 : 20))
{
int fps = vdp_pal ? (sdl_video.frames_rendered / 3) : sdl_video.frames_rendered;
sdl_sync.ticks = sdl_video.frames_rendered = 0;
2010-12-04 17:16:55 +00:00
sprintf(caption,"%d fps - %s ", fps, rominfo.international);
2010-01-27 07:12:26 +00:00
SDL_WM_SetCaption(caption, NULL);
}
2009-11-22 16:35:41 +00:00
return interval;
}
static int sdl_sync_init()
{
if(SDL_InitSubSystem(SDL_INIT_TIMER|SDL_INIT_EVENTTHREAD) < 0)
{
2009-11-22 16:35:41 +00:00
MessageBox(NULL, "SDL Timer initialization failed", "Error", 0);
return 0;
}
2009-11-22 16:35:41 +00:00
sdl_sync.sem_sync = SDL_CreateSemaphore(0);
2010-01-27 07:12:26 +00:00
sdl_sync.ticks = 0;
2009-11-22 16:35:41 +00:00
return 1;
}
static void sdl_sync_close()
{
if(sdl_sync.sem_sync)
SDL_DestroySemaphore(sdl_sync.sem_sync);
}
static int sdl_control_update(SDLKey keystate)
{
switch (keystate)
{
case SDLK_TAB:
{
2010-06-30 08:16:20 +00:00
system_init();
2009-11-22 16:35:41 +00:00
system_reset();
break;
}
case SDLK_F2:
{
if (fullscreen) fullscreen = 0;
else fullscreen = SDL_FULLSCREEN;
sdl_video.surf_screen = SDL_SetVideoMode(VIDEO_WIDTH, VIDEO_HEIGHT, 16, SDL_SWSURFACE | fullscreen);
break;
}
case SDLK_F3:
{
config.render ^=1;
break;
}
case SDLK_F4:
{
2010-06-30 08:16:20 +00:00
if (!turbo_mode) use_sound ^= 1;
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_F5:
{
log_error ^= 1;
break;
}
case SDLK_F6:
{
2010-06-30 08:16:20 +00:00
if (!use_sound) turbo_mode ^=1;
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_F7:
{
FILE *f = fopen("game.gpz","r+b");
if (f)
{
uint8 buf[STATE_SIZE];
fread(&buf, STATE_SIZE, 1, f);
state_load(buf);
fclose(f);
}
break;
}
case SDLK_F8:
{
FILE *f = fopen("game.gpz","w+b");
if (f)
{
uint8 buf[STATE_SIZE];
state_save(buf);
fwrite(&buf, STATE_SIZE, 1, f);
fclose(f);
}
break;
}
case SDLK_F9:
{
vdp_pal ^= 1;
/* save YM2612 context */
2010-01-27 07:12:26 +00:00
unsigned char *temp = malloc(YM2612GetContextSize());
if (temp)
memcpy(temp, YM2612GetContextPtr(), YM2612GetContextSize());
/* reinitialize all timings */
2010-01-27 07:12:26 +00:00
audio_init(snd.sample_rate, snd.frame_rate);
system_init();
/* restore YM2612 context */
2009-11-22 16:35:41 +00:00
if (temp)
{
YM2612Restore(temp);
free(temp);
}
2010-12-04 17:16:55 +00:00
/* reinitialize VC max value */
vc_max = 0xEA + 24*vdp_pal;
if (reg[1] & 8)
{
vc_max += (28 - 20*vdp_pal);
}
/* reinitialize display area */
bitmap.viewport.changed = 3;
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_F10:
{
2010-12-04 17:16:55 +00:00
gen_softreset(1);
gen_softreset(0);
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_F11:
{
2010-06-30 08:16:20 +00:00
config.overscan ^= 1;
2010-12-04 17:16:55 +00:00
bitmap.viewport.changed = 3;
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_F12:
{
2010-12-04 17:16:55 +00:00
while (input.dev[++joynum] == NO_DEVICE)
{
joynum = joynum % MAX_DEVICES;
}
joynum = joynum % MAX_DEVICES;
2009-11-22 16:35:41 +00:00
break;
}
case SDLK_ESCAPE:
{
return 0;
}
default:
break;
}
return 1;
}
int sdl_input_update(void)
{
uint8 *keystate = SDL_GetKeyState(NULL);
while (input.dev[joynum] == NO_DEVICE)
{
joynum ++;
if (joynum > MAX_DEVICES - 1) joynum = 0;
}
/* reset input */
input.pad[joynum] = 0;
/* keyboard */
if(keystate[SDLK_a]) input.pad[joynum] |= INPUT_A;
if(keystate[SDLK_s]) input.pad[joynum] |= INPUT_B;
if(keystate[SDLK_d]) input.pad[joynum] |= INPUT_C;
if(keystate[SDLK_f]) input.pad[joynum] |= INPUT_START;
if(keystate[SDLK_z]) input.pad[joynum] |= INPUT_X;
if(keystate[SDLK_x]) input.pad[joynum] |= INPUT_Y;
if(keystate[SDLK_c]) input.pad[joynum] |= INPUT_Z;
if(keystate[SDLK_v]) input.pad[joynum] |= INPUT_MODE;
switch (input.dev[joynum])
{
case DEVICE_LIGHTGUN:
{
/* get mouse (absolute values) */
int x,y;
int state = SDL_GetMouseState(&x,&y);
/* Calculate X Y axis values */
input.analog[joynum - 4][0] = (x * bitmap.viewport.w) / 640;
input.analog[joynum - 4][1] = (y * bitmap.viewport.h) / 480;
/* Map mouse buttons to player #1 inputs */
if(state & SDL_BUTTON_MMASK) input.pad[joynum] |= INPUT_C;
if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_B;
if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_A;
break;
}
case DEVICE_MOUSE:
{
/* get mouse (relative values) */
int x,y;
int state = SDL_GetRelativeMouseState(&x,&y);
/* Sega Mouse range is -256;+256 */
2010-12-04 17:16:55 +00:00
input.analog[2][0] = x * 2;
input.analog[2][1] = y * 2;
2009-11-22 16:35:41 +00:00
/* Vertical movement is upsidedown */
if (!config.invert_mouse)
input.analog[2][1] = 0 - input.analog[2][1];
2009-11-22 16:35:41 +00:00
/* Map mouse buttons to player #1 inputs */
if(state & SDL_BUTTON_MMASK) input.pad[joynum] |= INPUT_C;
if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_B;
if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_A;
break;
}
default:
{
if(keystate[SDLK_UP]) input.pad[joynum] |= INPUT_UP;
else
if(keystate[SDLK_DOWN]) input.pad[joynum] |= INPUT_DOWN;
if(keystate[SDLK_LEFT]) input.pad[joynum] |= INPUT_LEFT;
else
if(keystate[SDLK_RIGHT]) input.pad[joynum] |= INPUT_RIGHT;
break;
}
}
if (system_hw == SYSTEM_PICO)
{
/* get mouse (absolute values) */
int x,y;
int state = SDL_GetMouseState(&x,&y);
/* Calculate X Y axis values */
input.analog[0][0] = 0x3c + (x * (0x17c-0x03c+1)) / VIDEO_WIDTH;
input.analog[0][1] = 0x1fc + (y * (0x2f7-0x1fc+1)) / VIDEO_HEIGHT;
2009-11-22 16:35:41 +00:00
/* Map mouse buttons to player #1 inputs */
if(state & SDL_BUTTON_MMASK) pico_current++;
if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_B;
if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_A;
}
free (keystate);
return 1;
2008-10-13 16:07:02 +00:00
}
2008-08-07 12:43:17 +00:00
int main (int argc, char **argv)
{
int running = 1;
2008-08-07 12:43:17 +00:00
2008-08-21 20:34:00 +00:00
/* Print help if no game specified */
if(argc < 2)
{
char caption[256];
2008-08-25 20:58:05 +00:00
sprintf(caption, "Genesis Plus\\SDL by Charles MacDonald\nWWW: http://cgfm2.emuviews.com\nusage: %s gamename\n", argv[0]);
2008-08-21 20:34:00 +00:00
MessageBox(NULL, caption, "Information", 0);
exit(1);
}
2008-08-07 12:43:17 +00:00
/* set default config */
2009-11-22 16:35:41 +00:00
error_init();
set_config_defaults();
2009-11-22 16:35:41 +00:00
/* Load ROM file */
cart.rom = malloc(10*1024*1024);
memset(cart.rom, 0, 10*1024*1024);
2008-08-21 20:34:00 +00:00
if(!load_rom(argv[1]))
{
char caption[256];
sprintf(caption, "Error loading file `%s'.", argv[1]);
MessageBox(NULL, caption, "Error", 0);
exit(1);
}
/* load BIOS */
2008-10-13 16:07:02 +00:00
memset(bios_rom, 0, sizeof(bios_rom));
FILE *f = fopen(OS_ROM, "rb");
if (f!=NULL)
{
fread(&bios_rom, 0x800,1,f);
fclose(f);
int i;
for(i = 0; i < 0x800; i += 2)
{
uint8 temp = bios_rom[i];
bios_rom[i] = bios_rom[i+1];
bios_rom[i+1] = temp;
}
2010-06-30 08:16:20 +00:00
config.tmss |= 2;
2008-10-13 16:07:02 +00:00
}
2008-08-21 20:34:00 +00:00
2009-11-22 16:35:41 +00:00
/* initialize SDL */
if(SDL_Init(0) < 0)
{
char caption[256];
sprintf(caption, "SDL initialization failed");
MessageBox(NULL, caption, "Error", 0);
exit(1);
}
sdl_video_init();
2010-06-30 08:16:20 +00:00
if (use_sound) sdl_sound_init();
2009-11-22 16:35:41 +00:00
sdl_sync_init();
/* initialize Genesis virtual system */
SDL_LockSurface(sdl_video.surf_bitmap);
memset(&bitmap, 0, sizeof(t_bitmap));
bitmap.width = 720;
bitmap.height = 576;
bitmap.depth = 16;
bitmap.granularity = 2;
bitmap.pitch = (bitmap.width * bitmap.granularity);
bitmap.data = sdl_video.surf_bitmap->pixels;
SDL_UnlockSurface(sdl_video.surf_bitmap);
2010-06-30 08:16:20 +00:00
bitmap.viewport.changed = 3;
2009-11-22 16:35:41 +00:00
/* initialize emulation */
2010-12-04 17:16:55 +00:00
audio_init(SOUND_FREQUENCY, vdp_pal ? 50.0 : 60.0);
2008-08-21 20:34:00 +00:00
system_init();
2008-08-24 19:58:37 +00:00
/* load SRAM */
f = fopen("./game.srm", "rb");
if (f!=NULL)
{
2008-10-13 16:07:02 +00:00
fread(sram.sram,0x10000,1, f);
fclose(f);
}
/* reset emulation */
system_reset();
2008-08-21 20:34:00 +00:00
2010-06-30 08:16:20 +00:00
if(use_sound) SDL_PauseAudio(0);
2009-11-22 16:35:41 +00:00
/* 3 frames = 50 ms (60hz) or 60 ms (50hz) */
2009-11-22 16:35:41 +00:00
if(sdl_sync.sem_sync)
SDL_SetTimer(vdp_pal ? 60 : 50, sdl_sync_timer_callback);
2009-11-22 16:35:41 +00:00
/* emulation loop */
while(running)
{
SDL_Event event;
if (SDL_PollEvent(&event))
{
switch(event.type)
{
2009-11-22 16:35:41 +00:00
case SDL_QUIT:
running = 0;
break;
2009-11-22 16:35:41 +00:00
case SDL_KEYDOWN:
running = sdl_control_update(event.key.keysym.sym);
break;
}
}
2009-11-22 16:35:41 +00:00
sdl_video_update();
2010-01-27 07:12:26 +00:00
sdl_sound_update();
2008-08-07 12:43:17 +00:00
2009-11-22 16:35:41 +00:00
if(!turbo_mode && sdl_sync.sem_sync && sdl_video.frames_rendered % 3 == 0)
2008-08-24 19:58:37 +00:00
{
2009-11-22 16:35:41 +00:00
SDL_SemWait(sdl_sync.sem_sync);
}
2009-11-22 16:35:41 +00:00
2008-08-24 19:58:37 +00:00
}
/* save SRAM */
f = fopen("./game.srm", "wb");
if (f!=NULL)
{
fwrite(sram.sram,0x10000,1, f);
fclose(f);
}
2008-08-26 15:58:27 +00:00
system_shutdown();
audio_shutdown();
2008-08-24 19:58:37 +00:00
error_shutdown();
free(cart.rom);
2008-08-07 12:43:17 +00:00
2009-11-22 16:35:41 +00:00
sdl_video_close();
2010-06-30 08:16:20 +00:00
sdl_sound_close();
2009-11-22 16:35:41 +00:00
sdl_sync_close();
SDL_Quit();
2009-11-22 16:35:41 +00:00
return 0;
2008-08-07 12:43:17 +00:00
}