#ifdef __WIN32__
#include <windows.h>
#else
#define MessageBox(owner, text, caption, type) printf("%s: %s\n", caption, text)
#endif

#include "SDL.h"
#include "SDL_thread.h"

#include "shared.h"
#include "sms_ntsc.h"
#include "md_ntsc.h"
#include "utils.h"

#ifdef GCWZERO
#include <SDL_ttf.h>
#include <SDL_image.h>
#include <time.h>

static int do_once = 1;
static int gcw0_w = 320;
static int gcw0_h = 240;
static int gotomenu;
static int show_lightgun;
time_t current_time;

const char *cursor[4]=
{
    "./CLASSIC_01_RED.png", //doesn't flash (for epileptics it's default)
    "./CLASSIC_02.png", //square flashing red and white
    "./CLASSIC_01.png",
    "./SQUARE_02.png",
};

#define JOY_DEADZONE 1000
#endif

#ifdef GCWZERO
#define SOUND_FREQUENCY 44100
#else
#define SOUND_FREQUENCY 48000
#endif
#define SOUND_SAMPLES_SIZE 2048

#define VIDEO_WIDTH  320
#define VIDEO_HEIGHT 240

int joynum = 0;

int log_error   = 0;
int debug_on    = 0;
int turbo_mode  = 0;
int use_sound   = 1;
int fullscreen  = 1; /* SDL_FULLSCREEN */

char rom_filename[256];

/* sound */

struct
{
    char* current_pos;
    char* buffer;
    int current_emulated_samples;
} sdl_sound;


static uint8 brm_format[0x40] =
{
    0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x5f,0x00,0x00,0x00,0x00,0x40,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
    0x53,0x45,0x47,0x41,0x5f,0x43,0x44,0x5f,0x52,0x4f,0x4d,0x00,0x01,0x00,0x00,0x00,
    0x52,0x41,0x4d,0x5f,0x43,0x41,0x52,0x54,0x52,0x49,0x44,0x47,0x45,0x5f,0x5f,0x5f
};


static short soundframe[SOUND_SAMPLES_SIZE];

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;
    }
}

static int sdl_sound_init()
{
    int n;
    SDL_AudioSpec as_desired, as_obtained;

    if(SDL_Init(SDL_INIT_AUDIO) < 0)
    {
        MessageBox(NULL, "SDL Audio initialization failed", "Error", 0);
        return 0;
    }

    as_desired.freq     = SOUND_FREQUENCY;
    as_desired.format   = AUDIO_S16LSB;
    as_desired.channels = 2;
    as_desired.samples  = SOUND_SAMPLES_SIZE;
    as_desired.callback = sdl_sound_callback;

    if(SDL_OpenAudio(&as_desired, &as_obtained) == -1)
    {
        MessageBox(NULL, "SDL Audio open failed", "Error", 0);
        return 0;
    }
 
    if(as_desired.samples != as_obtained.samples)
    {
        MessageBox(NULL, "SDL Audio wrong setup", "Error", 0);
        return 0;
    }
 
    sdl_sound.current_emulated_samples = 0;
    n = SOUND_SAMPLES_SIZE * 2 * sizeof(short) * 20;
    sdl_sound.buffer = (char*)malloc(n);
    if(!sdl_sound.buffer)
    {
        MessageBox(NULL, "Can't allocate audio buffer", "Error", 0);
        return 0;
    }
    memset(sdl_sound.buffer, 0, n);
    sdl_sound.current_pos = sdl_sound.buffer;
    return 1;
}
 
static void sdl_sound_update(enabled)
{
    int size = audio_update(soundframe) * 2;
 
    if (enabled)
    {
        int i;
        short *out;
 
        SDL_LockAudio();
        out = (short*)sdl_sound.current_pos;
        for(i = 0; i < size; i++)
        {
            *out++ = soundframe[i];
        }
        sdl_sound.current_pos = (char*)out;
        sdl_sound.current_emulated_samples += size * sizeof(short);
        SDL_UnlockAudio();
    }
}
 
static void sdl_sound_close()
{
    SDL_PauseAudio(1);
    SDL_CloseAudio();
    if (sdl_sound.buffer)
        free(sdl_sound.buffer);
}

#ifdef GCWZERO //A-stick support
static void sdl_joystick_init()
{
    if(SDL_Init(SDL_INIT_JOYSTICK) < 0)
    {
        MessageBox(NULL, "SDL Joystick initialization failed", "Error", 0);
        return 0;
    } 
    else
        MessageBox(NULL, "SDL Joystick initialisation successful", "Success", 0);
    return 1;
}
#endif
 
/* video */
md_ntsc_t *md_ntsc;
sms_ntsc_t *sms_ntsc;
 
struct
{
    SDL_Surface* surf_screen;
    SDL_Surface* surf_bitmap;
    SDL_Rect srect;
    SDL_Rect drect;
    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;
    }
#ifdef GCWZERO
    sdl_video.surf_screen  = SDL_SetVideoMode(VIDEO_WIDTH, VIDEO_HEIGHT, 16, SDL_HWSURFACE |
#else
    sdl_video.surf_screen  = SDL_SetVideoMode(VIDEO_WIDTH, VIDEO_HEIGHT, 16, SDL_HWSURFACE | fullscreen |
#endif
#ifdef SDL_TRIPLEBUF
                             SDL_TRIPLEBUF
#else
                             SDL_DOUBLEBUF
#endif
                                             );
    sdl_video.surf_bitmap = SDL_CreateRGBSurface(SDL_HWSURFACE, 720, 576, 16, 0, 0, 0, 0);
    sdl_video.frames_rendered = 0;
    SDL_ShowCursor(0);
    return;
}
 
static void sdl_video_update()
{
static int test;
    if (system_hw == SYSTEM_MCD)
    {
#ifdef GCWZERO
        if (test >= config.gcw0_frameskip)  // >= in case frameskip has just been lowered
        {
            system_frame_scd(0); //render frame
            test = 0;
        } else {
            system_frame_scd(1); //skip frame render
            test ++;
        }
#else
        system_frame_scd(0);
#endif
    }
    else if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
#ifdef GCWZERO
    {
        if (test >= config.gcw0_frameskip) 
        {
            system_frame_gen(0);
            test = 0;
        } else {
            system_frame_gen(1);
            test ++;
        }
#else
        system_frame_gen(0);
#endif
    }
    else
    {
#ifdef GCWZERO
        if (test >= config.gcw0_frameskip) 
        {
            system_frame_sms(0);
            test = 0;
        } else {
            system_frame_sms(1);
            test ++;
        }
#else
        system_frame_sms(0);
#endif
    }
 
    /* viewport size changed */
    if(bitmap.viewport.changed & 1)
    {
        bitmap.viewport.changed &= ~1;
 
        /* source bitmap */
#ifdef GCWZERO //remove left bar bug with SMS roms
        if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) )
        {
            if (config.smsmaskleftbar)
                sdl_video.srect.x = 8;
            else
                sdl_video.srect.x = 0;

        }
        else
        {
            sdl_video.srect.x = 0;
        }
#else
            sdl_video.srect.x = 0;
#endif
        sdl_video.srect.y = 0;
        sdl_video.srect.w = bitmap.viewport.w+2*bitmap.viewport.x;
        sdl_video.srect.h = bitmap.viewport.h+2*bitmap.viewport.y;
        if (sdl_video.srect.w > VIDEO_WIDTH)
        {
#ifdef GCWZERO
            if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) )
            {
                if (config.smsmaskleftbar)
                    sdl_video.srect.x = (sdl_video.srect.w - VIDEO_WIDTH) / 2 + 8;
                else
                    sdl_video.srect.x = (sdl_video.srect.w - VIDEO_WIDTH) / 2;
                sdl_video.srect.w = VIDEO_WIDTH;
            }
            else
            {
                sdl_video.srect.x = (sdl_video.srect.w - VIDEO_WIDTH) / 2;
                sdl_video.srect.w = VIDEO_WIDTH;
            }
#else
            sdl_video.srect.x = (sdl_video.srect.w - VIDEO_WIDTH) / 2;
            sdl_video.srect.w = VIDEO_WIDTH;
#endif
        }
        if (sdl_video.srect.h > VIDEO_HEIGHT)
        {
            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;
 
        /* clear destination surface */
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
#ifdef GCWZERO //triple buffering so stop flicker
        SDL_Flip(sdl_video.surf_screen);
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
        SDL_Flip(sdl_video.surf_screen);
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
#endif
#if 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)
        {
            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;
            }
        }
        else
        {
            if (sms_ntsc)
            {
                free(sms_ntsc);
                sms_ntsc = NULL;
            }
 
            if (md_ntsc)
            {
                free(md_ntsc);
                md_ntsc = NULL;
            }
        }
#endif
    }
 
//DK IPU scaling for gg/sms roms
#ifdef GCWZERO
    if (config.gcw0_fullscreen)
    {
        if( (gcw0_w != sdl_video.drect.w) || (gcw0_h != sdl_video.drect.h) )
        {
            if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) )
            {
                if (config.smsmaskleftbar)
                {
                    sdl_video.srect.w = sdl_video.srect.w - 8;
                    sdl_video.drect.w = sdl_video.srect.w;
                    sdl_video.drect.x = 4;
                }
                else
                {
                    sdl_video.srect.w = sdl_video.srect.w ;
                    sdl_video.drect.w = sdl_video.srect.w;
                    sdl_video.drect.x = 0;
                }

            }
            else
            {
                sdl_video.drect.x = 0;
                sdl_video.drect.w = sdl_video.srect.w;
            }

            sdl_video.drect.h = sdl_video.srect.h;
            sdl_video.drect.y = 0;
            gcw0_w=sdl_video.drect.w;
            gcw0_h=sdl_video.drect.h;

            if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) )
            {
                sdl_video.surf_screen  = SDL_SetVideoMode(256,gcw0_h, 16, SDL_HWSURFACE |
#ifdef SDL_TRIPLEBUF
                                         SDL_TRIPLEBUF);
#else
                                         SDL_DOUBLEBUF);
#endif
            }
            else
            { 
            sdl_video.surf_screen  = SDL_SetVideoMode(gcw0_w,gcw0_h, 16, SDL_HWSURFACE |
#ifdef SDL_TRIPLEBUF
                                     SDL_TRIPLEBUF);
#else
                                     SDL_DOUBLEBUF);
#endif
            } 
        }
    }
    if (show_lightgun && !config.gcw0_fullscreen) // hack to remove cursor corruption of over game screen edge
    {
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
    }
#endif
    SDL_BlitSurface(sdl_video.surf_bitmap, &sdl_video.srect, sdl_video.surf_screen, &sdl_video.drect);
    //SDL_UpdateRect(sdl_video.surf_screen, 0, 0, 0, 0);
#ifdef GCWZERO
//  Add scanlines to Game Gear games if requested
    if ( (system_hw == SYSTEM_GG) && config.gg_scanlines)
    {
        SDL_Surface *scanlinesSurface;
        scanlinesSurface = IMG_Load("./scanlines.png");
        SDL_BlitSurface(scanlinesSurface, NULL, sdl_video.surf_screen, &sdl_video.drect);
	SDL_FreeSurface(scanlinesSurface);
    }
    if (show_lightgun)
    {
//      Remove previous cursor from black bars
        if (config.gcw0_fullscreen)
        {
            if (config.smsmaskleftbar)
            {
                if(system_hw == SYSTEM_SMS2)
                {
                SDL_Rect srect;
                srect.x = 0;
                srect.y = 0;
                srect.w = 4;
                srect.h = 192;
                SDL_FillRect(sdl_video.surf_screen, &srect, SDL_MapRGB(sdl_video.surf_screen->format, 0, 0, 0));
                srect.x = 252;
                SDL_FillRect(sdl_video.surf_screen, &srect, SDL_MapRGB(sdl_video.surf_screen->format, 0, 0, 0));
                }
            }
        }
        /* get mouse coordinates (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);

        SDL_Rect lrect;
        lrect.x = x-7;
        lrect.y = y-7;
        lrect.w = 15;
        lrect.h = 15;

        SDL_Surface *lightgunSurface;
        lightgunSurface = IMG_Load(cursor[config.cursor]);
        static lightgun_af = 0;
        SDL_Rect srect;
        srect.y = 0;
        srect.w = 15;
        srect.h = 15;

        //only show cursor if movement occurred within 3 seconds.
        time_t current_time2;
        current_time2 = time(NULL);

        if (lightgun_af >= 10)
        {
            srect.x = 0;
            if((current_time2 - current_time) < 3)
                SDL_BlitSurface(lightgunSurface, &srect, sdl_video.surf_screen, &lrect);
        } else
        {
            if (config.cursor != 0)
                srect.x = 15;
            else 
                srect.x = 0;
            if((current_time2 - current_time) < 3)
                SDL_BlitSurface(lightgunSurface, &srect, sdl_video.surf_screen, &lrect);
        }
        lightgun_af++;
        if (lightgun_af == 20) lightgun_af = 0;
        SDL_FreeSurface(lightgunSurface);
    } //show_lightgun
#endif

    SDL_Flip(sdl_video.surf_screen);
    ++sdl_video.frames_rendered;
}
 
static void sdl_video_close()
{
    if (sdl_video.surf_bitmap)
        SDL_FreeSurface(sdl_video.surf_bitmap);
    if (sdl_video.surf_screen)
        SDL_FreeSurface(sdl_video.surf_screen);
}
 
/* Timer Sync */
 
struct
{
    SDL_sem* sem_sync;
    unsigned ticks;
} sdl_sync;
 
static Uint32 sdl_sync_timer_callback(Uint32 interval)
{
#ifdef GCWZERO
    if (!gotomenu) 
    {
        SDL_SemPost(sdl_sync.sem_sync);
        sdl_sync.ticks++;
    }
#else
    SDL_SemPost(sdl_sync.sem_sync);
    sdl_sync.ticks++;
#endif
    if (sdl_sync.ticks == (vdp_pal ? 50 : 20))
    {
        SDL_Event event;
        SDL_UserEvent userevent;
 
        userevent.type = SDL_USEREVENT;
        userevent.code = vdp_pal ? (sdl_video.frames_rendered / 3) : sdl_video.frames_rendered;
        userevent.data1 = NULL;
        userevent.data2 = NULL;
        sdl_sync.ticks = sdl_video.frames_rendered = 0;
 
        event.type = SDL_USEREVENT;
        event.user = userevent;
 
        SDL_PushEvent(&event);
    }
    return interval;
}
 
static int sdl_sync_init()
{
    if(SDL_InitSubSystem(SDL_INIT_TIMER|SDL_INIT_EVENTTHREAD) < 0)
    {
        MessageBox(NULL, "SDL Timer initialization failed", "Error", 0);
        return 0;
    }
 
    sdl_sync.sem_sync = SDL_CreateSemaphore(0);
    sdl_sync.ticks = 0;
    return 1;
}
 
static void sdl_sync_close()
{
    if(sdl_sync.sem_sync)
        SDL_DestroySemaphore(sdl_sync.sem_sync);
}
 
static const uint16 vc_table[4][2] =
{
    /* NTSC, PAL */
    {0xDA , 0xF2},  /* Mode 4 (192 lines) */
    {0xEA , 0x102}, /* Mode 5 (224 lines) */
    {0xDA , 0xF2},  /* Mode 4 (192 lines) */
    {0x106, 0x10A}  /* Mode 5 (240 lines) */
};
 
static int sdl_control_update(SDLKey keystate)
{
    switch (keystate)
    {
#ifndef GCWZERO
    case SDLK_TAB:
    {
        system_reset();
        break;
    }
#endif
 
    case SDLK_F1:
    {
        if (SDL_ShowCursor(-1)) SDL_ShowCursor(0);
        else SDL_ShowCursor(1);
        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:
    {
        if (config.bios == 0) config.bios = 3;
        else if (config.bios == 3) config.bios = 1;
        break;
    }
 
    case SDLK_F4:
    {
        if (!turbo_mode) use_sound ^= 1;
        break;
    }
 
    case SDLK_F5:
    {
        log_error ^= 1;
        break;
    }
 
    case SDLK_F6:
    {
        if (!use_sound)
        {
            turbo_mode ^=1;
            sdl_sync.ticks = 0;
        }
        break;
    }
 
    case SDLK_F7:
    {
        char save_state_file[256];
        sprintf(save_state_file,"%s/%X.gp0", get_save_directory(), rominfo.realchecksum);
        FILE *f = fopen(save_state_file,"rb");
 
        if (f)
        {
            uint8 buf[STATE_SIZE];
            fread(&buf, STATE_SIZE, 1, f);
            state_load(buf);
            fclose(f);
        }
        break;
    }
 
    case SDLK_F8:
    {
        char save_state_file[256];
        sprintf(save_state_file,"%s/%X.gp0", get_save_directory(), rominfo.realchecksum);
        FILE *f = fopen(save_state_file,"wb");
        if (f)
        {
            uint8 buf[STATE_SIZE];
            int len = state_save(buf);
            fwrite(&buf, len, 1, f);
            fclose(f);
        }
        break;
    }
 
    case SDLK_F9:
    {
        config.region_detect = (config.region_detect + 1) % 5;
        get_region(0);
 
        /* framerate has changed, reinitialize audio timings */
        audio_init(snd.sample_rate, 0);
 
        /* system with region BIOS should be reinitialized */
        if ((system_hw == SYSTEM_MCD) || ((system_hw & SYSTEM_SMS) && (config.bios & 1)))
        {
            system_init();
            system_reset();
        }
        else
        {
            /* reinitialize I/O region register */
            if (system_hw == SYSTEM_MD)
            {
                io_reg[0x00] = 0x20 | region_code | (config.bios & 1);
            }
            else
            {
                io_reg[0x00] = 0x80 | (region_code >> 1);
            }
 
            /* reinitialize VDP */
            if (vdp_pal)
            {
                status |= 1;
                lines_per_frame = 313;
            }
            else
            {
                status &= ~1;
                lines_per_frame = 262;
            }
 
            /* reinitialize VC max value */
            switch (bitmap.viewport.h)
            {
            case 192:
                vc_max = vc_table[0][vdp_pal];
                break;
            case 224:
                vc_max = vc_table[1][vdp_pal];
                break;
            case 240:
                vc_max = vc_table[3][vdp_pal];
                break;
            }
        }
        break;
    }
 
    case SDLK_F10:
    {
        gen_reset(0);
        break;
    }
 
    case SDLK_F11:
    {
        config.overscan =  (config.overscan + 1) & 3;
        if ((system_hw == SYSTEM_GG) && !config.gg_extra)
        {
            bitmap.viewport.x = (config.overscan & 2) ? 14 : -48;
        }
        else
        {
            bitmap.viewport.x = (config.overscan & 2) * 7;
        }
        bitmap.viewport.changed = 3;
        break;
    }
    case SDLK_F12:
    {
        joynum = (joynum + 1) % MAX_DEVICES;
        while (input.dev[joynum] == NO_DEVICE)
        {
            joynum = (joynum + 1) % MAX_DEVICES;
        }
        break;
    }
 
    case SDLK_ESCAPE:
    {
#ifndef GCWZERO
        /* exit */
        return 0;
#endif
    }
 
    default:
        break;
    }
 
    return 1;
}
 
static void shutdown()
{
    FILE *fp;
 
    if (system_hw == SYSTEM_MCD)
    {
        /* save internal backup RAM (if formatted) */
        char brm_file[256];
        if (!memcmp(scd.bram + 0x2000 - 0x20, brm_format + 0x20, 0x20))
        {
			sprintf(brm_file,"%s/", get_save_directory(), "scd.brm");
			fp = fopen(brm_file, "wb");
            if (fp!=NULL)
            {
                fwrite(scd.bram, 0x2000, 1, fp);
                fclose(fp);
            }
        }
 
        /* save cartridge backup RAM (if formatted) */
        if (scd.cartridge.id)
        {
            if (!memcmp(scd.cartridge.area + scd.cartridge.mask + 1 - 0x20, brm_format + 0x20, 0x20))
            {
				sprintf(brm_file,"%s/", get_save_directory(), "cart.brm");
				fp = fopen(brm_file, "wb");
                if (fp!=NULL)
                {
                    fwrite(scd.cartridge.area, scd.cartridge.mask + 1, 1, fp);
                    fclose(fp);
                }
            }
        }
    }
 
    if (sram.on)
    {
        /* save SRAM */
        char save_file[256];
        if (rom_filename[0] != '\0') {
			sprintf(save_file,"%s/%s.srm", get_save_directory(), rom_filename);
			fp = fopen(save_file, "wb");
			if (fp!=NULL)
			{
				fwrite(sram.sram,0x10000,1, fp);
				fclose(fp);
			}
		}
    }
    audio_shutdown();
    error_shutdown();
 
    sdl_video_close();
    sdl_sound_close();
    sdl_sync_close();
    SDL_Quit();
}
 
#ifdef GCWZERO //menu!
static int gcw0menu(void)
{
    SDL_PauseAudio(1);

    /* display menu */
//  change video mode
    sdl_video.surf_screen  = SDL_SetVideoMode(320,240, 32, SDL_HWSURFACE |
#ifdef SDL_TRIPLEBUF
       SDL_TRIPLEBUF);
#else
       SDL_DOUBLEBUF);
#endif
//  blank screen
    SDL_FillRect(sdl_video.surf_screen, 0, 0);
 
    enum {MAINMENU = 0, GRAPHICS_OPTIONS = 1, REMAP_OPTIONS = 2, SAVE_STATE = 3, LOAD_STATE = 4, MISC_OPTIONS = 5};
    static int menustate  = MAINMENU;
//  Menu text
    const char *gcw0menu_mainlist[9]=
    {
        "Resume game",
        "Save state",
        "Load state",
        "Graphics options",
        "Remap buttons",
        "Misc. Options",
            
        "", //spacer
        "Reset",
        "Quit"
    };
    const char *gcw0menu_gfxlist[6]=
    {
        "Scaling",
        "Keep aspect ratio",
        "Scanlines (GG)",
        "Mask left bar (SMS)",
        "Frameskip",
        "Return to main menu",
    };
    const char *gcw0menu_numericlist[4]=
    {
        "0",
        "1",
        "2",
        "3",
    };
    const char *gcw0menu_onofflist[2]=
    {
        "Off",
        "On",
    };
    const char *gcw0menu_remapoptionslist[9]=
    {
        "A",
        "B",
        "C",
        "X",
        "Y",
        "Z",
        "Start",
        "Mode",
        "Return to main menu",
    };
    const char *gcw0menu_savestate[10]=
    {
        "Back to main menu",
        "Save state 1 (Quicksave)",
        "Save state 2",
        "Save state 3",
        "Save state 4",
        "Save state 5",
        "Save state 6",
        "Save state 7",
        "Save state 8",
        "Save state 9",
    };
    const char *gcw0menu_loadstate[10]=
    {
        "Back to main menu",
        "Load state 1 (Quickload)",
        "Load state 2",
        "Load state 3",
        "Load state 4",
        "Load state 5",
        "Load state 6",
        "Load state 7",
        "Load state 8",
        "Load state 9",
    };
    const char *gcw0menu_misc[7]=
    {
        "Back to main menu",
        "Resume on Save/Load",
        "A-stick",
        "Lock-on",
        "FM sound (SMS)",
        "Lightgun speed",
        "Lightgun Cursor",
    };

    const char *lock_on_desc[4]=
    {
		"             Off",
		"      Game Genie",
		"   Action Replay",
		"Sonic & Knuckles",
	};

//  start menu loop
    bitmap.viewport.changed=1; //change screen res if required
    while(gotomenu)
    {
//      set up menu surface
        SDL_Surface *menuSurface = NULL;
        menuSurface = SDL_CreateRGBSurface(SDL_HWSURFACE, 320, 240, 16, 0, 0, 0, 0);
 
//  identify system we are using to show correct background just cos we can :P
        if      (  system_hw == SYSTEM_PICO) //Sega Pico
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./PICO.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }
        else if ( (system_hw == SYSTEM_SG)      || (system_hw == SYSTEM_SGII) ) //SG-1000 I&II
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./SG1000.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }
        else if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || ( system_hw == SYSTEM_GGMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) ) //Mark III & Sega Master System I&II & Megadrive with power base converter
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./SMS.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }
        else if (system_hw == SYSTEM_GG)
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./GG.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }
        else if (  system_hw == SYSTEM_MD)   //Megadrive
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./MD.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }
        else if (  system_hw == SYSTEM_MCD)  //MegaCD
        {
            SDL_Surface *tempbgSurface;
            SDL_Surface *bgSurface;
            tempbgSurface = IMG_Load( "./MCD.png" );
            bgSurface = SDL_DisplayFormat( tempbgSurface );
            SDL_BlitSurface(bgSurface, NULL, menuSurface, NULL);
            SDL_FreeSurface(tempbgSurface);
            SDL_FreeSurface(bgSurface);
        }

//      show menu
        TTF_Init();
        TTF_Font *ttffont = NULL;
        SDL_Color text_color = {180, 180, 180};
        SDL_Color selected_text_color = {23, 86, 155}; //selected colour = Sega blue ;)
        SDL_Surface *textSurface;

        int i;
        static int selectedoption = 0;

//      Fill menu box
        SDL_Surface *MenuBackground;
        if (menustate == REMAP_OPTIONS)
        {
            MenuBackground = SDL_CreateRGBSurface(SDL_HWSURFACE, 320, 185, 16, 0, 0, 0, 0);
            SDL_Rect rect;
            rect.x = 0;
            rect.y = 35;
            rect.w = 320;
            rect.h = 185;
            SDL_FillRect(MenuBackground, 0, 0);
            SDL_SetAlpha(MenuBackground, SDL_SRCALPHA, 50);
            SDL_BlitSurface(MenuBackground, NULL, menuSurface, &rect);
            SDL_FreeSurface(MenuBackground);
        }
        else
        {
            MenuBackground = SDL_CreateRGBSurface(SDL_HWSURFACE, 180, 185, 16, 0, 0, 0, 0);
            SDL_Rect rect;
            rect.x = 60;
            rect.y = 35;
            rect.w = 180;
            rect.h = 185;
            SDL_FillRect(MenuBackground, 0, 0);
            SDL_SetAlpha(MenuBackground, SDL_SRCALPHA, 50);
            SDL_BlitSurface(MenuBackground, NULL, menuSurface, &rect);
            SDL_FreeSurface(MenuBackground);
        }

//	Show title
        ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
        SDL_Rect destination;
        destination.x = 80;
        destination.y = 40;
        destination.w = 100;
        destination.h = 50;
        textSurface = TTF_RenderText_Solid(ttffont, "Genesis Plus GX", text_color);
        SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
        SDL_FreeSurface(textSurface);
        TTF_CloseFont (ttffont);
 
        if (menustate == MAINMENU)
        {
          //there's no need to open/close font each cycle :P
            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            for(i=0; i<9; i++)
            {
                SDL_Rect destination;
                destination.x = 80;
                destination.y = 70+(15*i);
                destination.w = 100;
                destination.h = 50;
                if (i == selectedoption)
                    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_mainlist[i], selected_text_color);
                else
                    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_mainlist[i], text_color);
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            TTF_CloseFont (ttffont);
        }
        else if (menustate == GRAPHICS_OPTIONS)
        {
			ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            for(i=0; i<6; i++)
            {
                SDL_Rect destination;
                destination.x = 80;
                destination.y = 70+(15*i);
                destination.w = 100;
                destination.h = 50;
                if ((i+10) == selectedoption)
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_gfxlist[i], selected_text_color);
		        else
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_gfxlist[i], text_color);
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            /* Display On/Off */
            SDL_Rect destination;
	        destination.x = 220;
	        destination.w = 100; 
	        destination.h = 50;
//          Scaling
            destination.y = 70+(15*0);
            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.gcw0_fullscreen], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
	        SDL_FreeSurface(textSurface);
//          Aspect ratio
            destination.y = 70+(15*1);
    	    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.keepaspectratio], selected_text_color);
	        SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
	        SDL_FreeSurface(textSurface);
//          Scanlines
            destination.y = 70+(15*2);
    	    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.gg_scanlines], selected_text_color);
	        SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
	        SDL_FreeSurface(textSurface);
//          Mask left bar
            destination.y = 70+(15*3);
    	    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.smsmaskleftbar], selected_text_color);
	        SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
	        SDL_FreeSurface(textSurface);
//          Frameskip
            destination.y = 70+(15*4);
    	    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_numericlist[config.gcw0_frameskip], selected_text_color);
	        SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
	        SDL_FreeSurface(textSurface);

            TTF_CloseFont (ttffont);

        }
        else if (menustate == REMAP_OPTIONS)
        {
            char* remap_text[256];
            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            sprintf(remap_text, "%s%25s", "GenPlus", "GCW-Zero");
            SDL_Rect destination = {30, 60, 100, 50};
            textSurface = TTF_RenderText_Solid(ttffont, remap_text, text_color);
            SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);

            for(i=0; i < 9; i++)
            {
                if (i < 8)
                {
                    sprintf(remap_text, "%-5s                   %-7s", gcw0menu_remapoptionslist[i], gcw0_get_key_name(config.buttons[i]));
                } else
                {
                    sprintf(remap_text, gcw0menu_remapoptionslist[i]); // for return option
                }
                SDL_Rect destination = {30, 80 + (15 * i), 100, 50};
                if ((i+20) == selectedoption)
                {
                    textSurface = TTF_RenderText_Solid(ttffont, remap_text, selected_text_color);
                } else
                {
                    textSurface = TTF_RenderText_Solid(ttffont, remap_text, text_color);
                }
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            TTF_CloseFont (ttffont);
        }
        else if (menustate == SAVE_STATE)
        {
          //Show saved BMP as background if available
            SDL_Surface* screenshot;
            char load_state_screenshot[256];
            sprintf(load_state_screenshot,"%s/%s.%d.bmp", get_save_directory(), rom_filename, selectedoption-30);
            screenshot = SDL_LoadBMP(load_state_screenshot);
            if (screenshot)
            {
                SDL_Rect destination;
                destination.x = (320 - screenshot->w) / 2;
                destination.y = (240 - screenshot->h) / 2;
                destination.w = 320;
                destination.h = 240;
                SDL_BlitSurface(screenshot, NULL, menuSurface, &destination);

//              Fill menu box
                SDL_Surface *MenuBackground = SDL_CreateRGBSurface(SDL_HWSURFACE, 180, 185, 16, 0, 0, 0, 0);
                SDL_Rect rect;
                rect.x = 60;
                rect.y = 35;
                rect.w = 180;
                rect.h = 185;
                SDL_FillRect(MenuBackground, 0, 0);
                SDL_SetAlpha(MenuBackground, SDL_SRCALPHA, 180);
                SDL_BlitSurface(MenuBackground, NULL, menuSurface, &rect);
                SDL_FreeSurface(MenuBackground);
            }
            SDL_FreeSurface(screenshot);

//          Show title
            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            SDL_Rect destination;
            destination.x = 80;
            destination.y = 40;
            destination.w = 100;
            destination.h = 50;
            textSurface = TTF_RenderText_Solid(ttffont, "Genesis Plus GX", text_color);
            SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
            TTF_CloseFont (ttffont);

            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            for(i=0; i<10; i++)
            {
                SDL_Rect destination;
                destination.x = 80;
                destination.y = 70+(15*i);
                destination.w = 100;
                destination.h = 50;
                if ((i+30) == selectedoption)
                    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_savestate[i], selected_text_color);
	        else
                    textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_savestate[i], text_color);
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            TTF_CloseFont (ttffont);
        }
        else if (menustate == LOAD_STATE)
        {
          //Show saved BMP as background if available
            SDL_Surface* screenshot;
            char load_state_screenshot[256];
            sprintf(load_state_screenshot,"%s/%s.%d.bmp", get_save_directory(), rom_filename, selectedoption-40);
            screenshot = SDL_LoadBMP(load_state_screenshot);
            if (screenshot)
            {
                SDL_Rect destination;
                destination.x = (320 - screenshot->w) / 2;
                destination.y = (240 - screenshot->h) / 2;
                destination.w = 320;
                destination.h = 240;
                SDL_BlitSurface(screenshot, NULL, menuSurface, &destination);

//              Fill menu box
                SDL_Surface *MenuBackground = SDL_CreateRGBSurface(SDL_HWSURFACE, 180, 185, 16, 0, 0, 0, 0);
                SDL_Rect rect;
                rect.x = 60;
                rect.y = 35;
                rect.w = 180;
                rect.h = 185;
                SDL_FillRect(MenuBackground, 0, 0);
                SDL_SetAlpha(MenuBackground, SDL_SRCALPHA, 180);
                SDL_BlitSurface(MenuBackground, NULL, menuSurface, &rect);
                SDL_FreeSurface(MenuBackground);
            }
            SDL_FreeSurface(screenshot);

//          Show title
            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            SDL_Rect destination;
            destination.x = 80;
            destination.y = 40;
            destination.w = 100;
            destination.h = 50;
            textSurface = TTF_RenderText_Solid(ttffont, "Genesis Plus GX", text_color);
            SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
            TTF_CloseFont (ttffont);

            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            for(i=0; i<10; i++)
            {
                SDL_Rect destination;
                destination.x = 80;
                destination.y = 70+(15*i);
                destination.w = 100;
                destination.h = 50;
                if ((i+40) == selectedoption)
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_loadstate[i], selected_text_color);
		        else
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_loadstate[i], text_color);
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            TTF_CloseFont (ttffont);
        }
        else if (menustate == MISC_OPTIONS)
        {
            ttffont = TTF_OpenFont("./ProggyTiny.ttf", 16);
            for(i=0; i<7; i++)
            {
                SDL_Rect destination;
                destination.x = 80;
                destination.y = 70+(15*i);
                destination.w = 100;
                destination.h = 50;
                if ((i+50) == selectedoption)
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_misc[i], selected_text_color);
		        else
		            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_misc[i], text_color);
                SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
                SDL_FreeSurface(textSurface);
            }
            /* Display On/Off */
            SDL_Rect destination;
            destination.x = 220;
            destination.w = 100; 
            destination.h = 50;
//          Save/load autoresume
            destination.y = 70+(15*1);
            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.sl_autoresume], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
//          A-stick
            destination.y = 70+(15*2);
            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.a_stick], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
            /* Display Lock-on Types */
            destination.x = 142;
            destination.y = 70+(15*3);
            textSurface = TTF_RenderText_Solid(ttffont, lock_on_desc[config.lock_on], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
//          FM sound(SMS)
            destination.x = 220;
            destination.y = 70+(15*4);
            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_onofflist[config.ym2413], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
//          Lightgun speed
            destination.x = 220;
            destination.y = 70+(15*5);
            textSurface = TTF_RenderText_Solid(ttffont, gcw0menu_numericlist[config.lightgun_speed], selected_text_color);
    	    SDL_BlitSurface(textSurface, NULL, menuSurface, &destination);
            SDL_FreeSurface(textSurface);
//          Lightgun Cursor
            destination.y = 70+(15*6);
            SDL_Surface *lightgunSurface;
            lightgunSurface = IMG_Load(cursor[config.cursor]);
            static lightgun_af_demo = 0;
            SDL_Rect srect;
            srect.x = 0;
            srect.y = 0;
            srect.w = 15;
            srect.h = 15;
            if (lightgun_af_demo >= 10 && config.cursor != 0)
            {
                srect.x = 15;
            }
            lightgun_af_demo++;
            if (lightgun_af_demo == 20) lightgun_af_demo = 0;
            SDL_BlitSurface(lightgunSurface, &srect, menuSurface, &destination);
            SDL_FreeSurface(lightgunSurface);

            TTF_CloseFont (ttffont);

        }

//TODO other menu's go here
 
 
        /* Update display */
        SDL_Rect dest;
        dest.w = 320;
        dest.h = 240;
        dest.x = 0;
        dest.y = 0;
        SDL_BlitSurface(menuSurface, NULL, sdl_video.surf_screen, &dest);
        SDL_FreeSurface(menuSurface);
        SDL_Flip(sdl_video.surf_screen);
        /* Check for user input */
        SDL_EnableKeyRepeat(0,0);
        static int keyheld=0;
        SDL_Event event;
        while(SDL_PollEvent(&event))
        {
            switch(event.type)
            {
            case SDL_KEYDOWN:
                sdl_control_update(event.key.keysym.sym);
                break;
            case SDL_KEYUP:
                keyheld = 0;
                break;
            default:
                break;
            }
        }
        if (event.type == SDL_KEYDOWN && !keyheld)
        {
            keyheld++;
            uint8 *keystate2;
            keystate2 = SDL_GetKeyState(NULL);
            if(keystate2[SDLK_DOWN])
            {
                if        (selectedoption >  9 && selectedoption < 20) //graphics menu
                {
                    selectedoption++;
                    if (selectedoption == 16) selectedoption = 10;
                } 
                else if (selectedoption > 19 && selectedoption < 30) //remap menu
                {
                    selectedoption++;
                    if (selectedoption == 29)    selectedoption = 20;
                } 
                else if (selectedoption > 29 && selectedoption < 40) //save menu
                {
                    selectedoption++;
                    if (selectedoption == 40)    selectedoption = 30;
    	        } 
                else if (selectedoption > 39 && selectedoption < 50) //load menu
    	        {
                    selectedoption++;
                    if (selectedoption == 50)    selectedoption = 40;
                } 
                else if (selectedoption > 49 && selectedoption < 60) //misc menu
    	        {
                    selectedoption++;
                    if (selectedoption == 57)    selectedoption = 50;
                } 
                else  //main menu
                {
                    selectedoption++;
                    if (selectedoption == 6)     selectedoption = 7;
	            if (selectedoption > 8)	     selectedoption=0;
    	        }
                SDL_Delay(100);
    	    }
            else if(keystate2[SDLK_UP])
            {
                if        (selectedoption > 9  && selectedoption < 20) //graphics menu
                {
                    selectedoption--;
                    if (selectedoption == 9)     selectedoption = 15;
    	        }
                else if (selectedoption > 19 && selectedoption < 30) //remap menu
    	        {
                    selectedoption--;
                    if (selectedoption == 19)    selectedoption = 28;
                }
                else if (selectedoption > 29 && selectedoption < 40) //save menu
                {
                    selectedoption--;
                    if (selectedoption == 29)    selectedoption = 39;
                }
                else if (selectedoption > 39 && selectedoption < 50) //load menu
                {
                    selectedoption--;
                    if (selectedoption == 39)    selectedoption = 49;
                }
                else if (selectedoption > 49 && selectedoption < 60) //misc menu
                {
                    selectedoption--;
                    if (selectedoption == 49)    selectedoption = 56;
                }
                else
    	        { //main menu
    	            if (!selectedoption)         selectedoption = 8;
  	            else                         selectedoption--;
	            if (selectedoption == 6)     selectedoption = 5;
    	        }
                SDL_Delay(100);
            }
	    else if(keystate2[SDLK_LALT] && menustate != REMAP_OPTIONS) //back to last menu or quit menu
            {
                if (menustate == GRAPHICS_OPTIONS)
                {
                    menustate = MAINMENU;
                    selectedoption = 3;
                    SDL_Delay(130);
                }
                else if (menustate == SAVE_STATE)
                {
                    menustate = MAINMENU;
                    selectedoption = 1;
                    SDL_Delay(130);
                }
                else if (menustate == LOAD_STATE)
                {
                    menustate = MAINMENU;
                    selectedoption = 2;
                    SDL_Delay(130);
                }
                else if (menustate == MISC_OPTIONS)
                {
                    menustate = MAINMENU;
                    selectedoption = 5;
                    SDL_Delay(130);
                }
                else if (menustate == MAINMENU)
                {
                    gotomenu=0;
                    selectedoption=0;
                    SDL_Delay(130);
	            break;
                }
            }
            else if(keystate2[SDLK_LCTRL] && menustate != REMAP_OPTIONS)
            {
                if (selectedoption == 0) 
	        { //Resume
	            gotomenu=0;
                    selectedoption=0;
                    SDL_Delay(130);
	            break;
                }
                else if (selectedoption == 1)   //Save
                {
                    menustate = SAVE_STATE;
                    selectedoption = 30;
                    SDL_Delay(130);
                }
                else if (selectedoption == 2)   //Load
                {
                    menustate = LOAD_STATE;
                    selectedoption = 40;
                    SDL_Delay(130);
                }
                else if (selectedoption == 3)   //Graphics
                {
                    menustate = GRAPHICS_OPTIONS;
                    selectedoption = 10;
	            SDL_Delay(200);
                }
                else if (selectedoption == 4)   //Remap
                {
                    menustate = REMAP_OPTIONS;
                    selectedoption=20;
                    SDL_Delay(200);
                }
                else if (selectedoption == 5)   //Misc.
                {
                    menustate = MISC_OPTIONS;
                    selectedoption=50;
                    SDL_Delay(200);
                }
                else if (selectedoption == 7)   //Reset
                {
                    gotomenu = 0;
                    selectedoption=0;
                    system_reset();
                    SDL_Delay(130);
                    break;
                }
                else if (selectedoption == 8)   //Quit
                {
                    exit(0);
                    SDL_Delay(130);
                    break;
                }
                else if (selectedoption == 10) 
                { //Scaling
                    config.gcw0_fullscreen = !config.gcw0_fullscreen;
                    SDL_Delay(130);
                    config_save();
                }
                else if (selectedoption == 11) 
                { //Keep aspect ratio
                    SDL_Delay(130);
                    config.keepaspectratio = !config.keepaspectratio;
                    config_save();
                    do_once = 1;
       	        }
                else if (selectedoption == 12) 
                { //Scanlines (GG)
                    SDL_Delay(130);
                    config.gg_scanlines = !config.gg_scanlines;
                    config_save();
                }
                else if (selectedoption == 13) 
                { //Mask left bar
                    SDL_Delay(130);
                    config.smsmaskleftbar = !config.smsmaskleftbar;
                    config_save();
                }
                else if (selectedoption == 14) 
                { //Frameskip
                    SDL_Delay(130);
                    config.gcw0_frameskip ++;
                    if (config.gcw0_frameskip == 4) config.gcw0_frameskip = 0;
                    config_save();
                }
                else if (selectedoption == 15) 
                { //Back to main menu
                    menustate = MAINMENU;
                    selectedoption = 3;
                    SDL_Delay(130);
                }
                else if (selectedoption == 30)
                {
                  //Return to main menu
                    menustate = MAINMENU;
                    selectedoption = 1;
                    SDL_Delay(130);
                }
                else if (selectedoption > 30 && selectedoption < 40)
                {
                  //save to selected savestate
                    char save_state_file[256];
                    sprintf(save_state_file,"%s/%s.gp%d", get_save_directory(), rom_filename, selectedoption-30);
                    FILE *f = fopen(save_state_file,"wb");
                    if (f)
                    {
                        uint8 buf[STATE_SIZE];
                        int len = state_save(buf);
                        fwrite(&buf, len, 1, f);
                        fclose(f);
                    }

                  //Save BMP screenshot
                    char save_state_screenshot[256];
                    sprintf(save_state_screenshot,"%s/%s.%d.bmp", get_save_directory(), rom_filename, selectedoption-30);
                    SDL_Surface* screenshot;
                    if (!config.gcw0_fullscreen)
                    {
                        screenshot = SDL_CreateRGBSurface(SDL_HWSURFACE, sdl_video.srect.w, sdl_video.srect.h, 16, 0, 0, 0, 0);
                        SDL_Rect temp;
                        temp.x = 0;
                        temp.y = 0;
                        temp.w = sdl_video.srect.w;
                        temp.h = sdl_video.srect.h;

                        SDL_BlitSurface(sdl_video.surf_bitmap, &temp, screenshot, &temp);
                        SDL_SaveBMP(screenshot, save_state_screenshot);
                        SDL_FreeSurface(screenshot);
                    }
                    else
                    {
                        screenshot = SDL_CreateRGBSurface(SDL_HWSURFACE, gcw0_w, gcw0_h, 16, 0, 0, 0, 0);
                        SDL_Rect temp;
                        temp.x = 0;
                        temp.y = 0;
                        temp.w = gcw0_w;
                        temp.h = gcw0_h;

                        SDL_BlitSurface(sdl_video.surf_bitmap, &temp, screenshot, &temp);
                        SDL_SaveBMP(screenshot, save_state_screenshot);
                        SDL_FreeSurface(screenshot);
                    }

                    SDL_Delay(250);
                    if (config.sl_autoresume)
                    {
                        menustate = MAINMENU;
                        selectedoption = 0;
                        gotomenu = 0;
                        break;
                    }
                }
                else if (selectedoption == 40)
                {
                  //return to main menu
                    menustate = MAINMENU;
                    selectedoption = 2;
                    SDL_Delay(130);
                }
                else if (selectedoption > 40 && selectedoption < 50)
                {
                  //load selected loadstate
                    char save_state_file[256];
                    sprintf(save_state_file,"%s/%s.gp%d", get_save_directory(), rom_filename, selectedoption-40 );
                    FILE *f = fopen(save_state_file,"rb");
                    if (f)
                    {
                        uint8 buf[STATE_SIZE];
                        fread(&buf, STATE_SIZE, 1, f);
                        state_load(buf);
                        fclose(f);
                    }
                    if (config.sl_autoresume)
                    {
                        gotomenu = 0;
                        menustate = MAINMENU;
                        selectedoption = 0;
                        SDL_Delay(250);
                        break;
                    }
                }
                else if (selectedoption == 50)
                {
                  //return to main menu
                    menustate = MAINMENU;
                    selectedoption = 5;
                    SDL_Delay(130);
                }
                else if (selectedoption == 51)
                {
                  //toggle auto resume when save/loading
                    config.sl_autoresume=!config.sl_autoresume;
                    config_save();
                    SDL_Delay(130);
                }
                else if (selectedoption == 52)
                {
                  //toggle A-Stick
                    config.a_stick=!config.a_stick;
                    config_save();
                    SDL_Delay(130);
                }
                else if (selectedoption == 53)
                {
                    config.lock_on = (++config.lock_on == 4)? 0 : config.lock_on;
                    config_save();
                    SDL_Delay(130);
                }
                else if (selectedoption == 54)
                {
                    config.ym2413 = !config.ym2413;
                    config_save();
                    SDL_Delay(130);
                }
                else if (selectedoption == 55)
                {
                    config.lightgun_speed++;
                    if (config.lightgun_speed == 4)
                        config.lightgun_speed = 1;
                    config_save();
                    SDL_Delay(130);
                }
                else if (selectedoption == 56)
                {
                    config.cursor++;
                    if (config.cursor == 4)
                        config.cursor = 0;
                    config_save();
                    SDL_Delay(130);
                }

            }
            else if(menustate == REMAP_OPTIONS)
            {// REMAP_OPTIONS needs to capture all input
                SDLKey pressed_key = 0;

                if (keystate2[SDLK_RETURN])
                    pressed_key = SDLK_RETURN;
                else if (keystate2[SDLK_LCTRL])
                    pressed_key = SDLK_LCTRL;
                else if (keystate2[SDLK_LALT]) 
                    pressed_key = SDLK_LALT;
                else if (keystate2[SDLK_LSHIFT]) 
                    pressed_key = SDLK_LSHIFT;
                else if (keystate2[SDLK_SPACE]) 
                    pressed_key = SDLK_SPACE;
                else if (keystate2[SDLK_TAB]) 
                    pressed_key = SDLK_TAB;
                else if (keystate2[SDLK_BACKSPACE]) 
                    pressed_key = SDLK_BACKSPACE;
                else if (keystate2[SDLK_ESCAPE]) 
                    pressed_key = SDLK_ESCAPE;

                if (pressed_key)
                {
                    if (selectedoption == 20)
                    {
                      //button a remap
                        config.buttons[A] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 21)
                    {
                      //button b remap
                        config.buttons[B] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 22)
                    {
                      //button c remap
                        config.buttons[C] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 23)
                    {
                      //button x remap
                        config.buttons[X] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 24)
                    {
                      //button y remap
                        config.buttons[Y] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 25)
                    {
                      //button z remap
                        config.buttons[Z] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 26)
                    {
                      //button start remap
                        config.buttons[START] = (pressed_key==SDLK_ESCAPE)? 0: pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 27)
                    {
                      //button mode remap
                        config.buttons[MODE] = pressed_key;
                        config_save();
                        SDL_Delay(130);
                        selectedoption++;
                    }
                    else if (selectedoption == 28)
                    {
                      //return to main menu
                        menustate = MAINMENU;
                        selectedoption = 4;
                        SDL_Delay(130);
                    }
                }
            }
        }
    }//menu loop
    SDL_PauseAudio(0);

    if(config.gcw0_fullscreen) {
        if ( (system_hw == SYSTEM_MARKIII) || (system_hw == SYSTEM_SMS) || (system_hw == SYSTEM_SMS2) || (system_hw == SYSTEM_PBC) )
        {
            gcw0_w=sdl_video.drect.w;
            gcw0_h=sdl_video.drect.h;
            sdl_video.surf_screen  = SDL_SetVideoMode(256,gcw0_h, 16, SDL_HWSURFACE |
#ifdef SDL_TRIPLEBUF
                                     SDL_TRIPLEBUF);
#else
                                     SDL_DOUBLEBUF);
#endif
        }
        else
        { 
            sdl_video.drect.w = sdl_video.srect.w;
            sdl_video.drect.h = sdl_video.srect.h;
            sdl_video.drect.x = 0;
            sdl_video.drect.y = 0;
            gcw0_w=sdl_video.drect.w;
            gcw0_h=sdl_video.drect.h;
            sdl_video.surf_screen  = SDL_SetVideoMode(gcw0_w,gcw0_h, 16, SDL_HWSURFACE |
#ifdef SDL_TRIPLEBUF
                                     SDL_TRIPLEBUF);
#else
                                     SDL_DOUBLEBUF);
#endif
        } 
    } else {
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
        SDL_Flip(sdl_video.surf_screen);
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
        SDL_Flip(sdl_video.surf_screen);
        SDL_FillRect(sdl_video.surf_screen, 0, 0);
        SDL_Flip(sdl_video.surf_screen);
    }
    return 1;
}
#endif
 
int sdl_input_update(void)
{
    uint8 *keystate = SDL_GetKeyState(NULL);

    /* reset input */
    input.pad[joynum] = 0;
    if(show_lightgun)
        input.pad[4] = 0; //player2:
    switch (input.dev[4])
    {
    case DEVICE_LIGHTGUN:
        show_lightgun = 1;
        /* get mouse coordinates (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
 
        if (config.gcw0_fullscreen)
        {
            input.analog[4][0] =  x;
            input.analog[4][1] =  y;
        } else
        {
            input.analog[4][0] =  x - (VIDEO_WIDTH-bitmap.viewport.w)/2;
            input.analog[4][1] =  y - (VIDEO_HEIGHT-bitmap.viewport.h)/2;
        } 
        if (config.smsmaskleftbar) x += 8;
        /* TRIGGER, B, C (Menacer only), START (Menacer & Justifier only) */
        if(keystate[SDLK_ESCAPE])  input.pad[4] |= INPUT_START;
    default:
        break;
    }
    switch (input.dev[joynum])
    {
    case DEVICE_LIGHTGUN:
    {

#ifdef GCWZERO
        show_lightgun = 2;
        /* get mouse coordinates (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
 
        if (config.gcw0_fullscreen)
        {
            input.analog[0][0] =  x;
            input.analog[0][1] =  y;
        } else
        {
            input.analog[0][0] =  x - (VIDEO_WIDTH-bitmap.viewport.w)/2;
            input.analog[0][1] =  y - (VIDEO_HEIGHT-bitmap.viewport.h)/2;
        } 
        if (config.smsmaskleftbar) x += 8;
        /* TRIGGER, B, C (Menacer only), START (Menacer & Justifier only) */
        if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_A;
        if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_B;
        if(state & SDL_BUTTON_MMASK) input.pad[joynum] |= INPUT_C;
        if(keystate[SDLK_ESCAPE])  input.pad[0] |= INPUT_START;
#else
        /* get mouse coordinates (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
 
        /* X axis */
	        input.analog[joynum][0] =  x - (VIDEO_WIDTH-bitmap.viewport.w)/2;
 
        /* Y axis */
        input.analog[joynum][1] =  y - (VIDEO_HEIGHT-bitmap.viewport.h)/2;
 
        /* TRIGGER, B, C (Menacer only), START (Menacer & Justifier only) */
        if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_A;
        if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_B;
        if(state & SDL_BUTTON_MMASK) input.pad[joynum] |= INPUT_C;
        if(keystate[SDLK_f])  input.pad[joynum] |= INPUT_START;
        break;
#endif
    }
#ifndef GCWZERO
    case DEVICE_PADDLE:
    {
        /* get mouse (absolute values) */
        int x;
        int state = SDL_GetMouseState(&x, NULL);
 
        /* Range is [0;256], 128 being middle position */
        input.analog[joynum][0] = x * 256 /VIDEO_WIDTH;
 
        /* Button I -> 0 0 0 0 0 0 0 I*/
        if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_B;
 
        break;
    }
 
    case DEVICE_SPORTSPAD:
    {
        /* get mouse (relative values) */
        int x,y;
        int state = SDL_GetRelativeMouseState(&x,&y);
 
        /* Range is [0;256] */
        input.analog[joynum][0] = (unsigned char)(-x & 0xFF);
        input.analog[joynum][1] = (unsigned char)(-y & 0xFF);
 
        /* Buttons I & II -> 0 0 0 0 0 0 II I*/
        if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_B;
        if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_C;
 
        break;
    }
 
    case DEVICE_MOUSE:
    {
    SDL_ShowCursor(1);
        /* get mouse (relative values) */
        int x,y;
        int state = SDL_GetRelativeMouseState(&x,&y);
 
        /* Sega Mouse range is [-256;+256] */
        input.analog[joynum][0] = x * 2;
        input.analog[joynum][1] = y * 2;
 
        /* Vertical movement is upsidedown */
        if (!config.invert_mouse)
            input.analog[joynum][1] = 0 - input.analog[joynum][1];
 
        /* Start,Left,Right,Middle buttons -> 0 0 0 0 START MIDDLE RIGHT LEFT */
        if(state & SDL_BUTTON_LMASK) input.pad[joynum] |= INPUT_B;
        if(state & SDL_BUTTON_RMASK) input.pad[joynum] |= INPUT_C;
        if(state & SDL_BUTTON_MMASK) input.pad[joynum] |= INPUT_A;
        if(keystate[SDLK_f])  input.pad[joynum] |= INPUT_START;
 
        break;
    }
 
    case DEVICE_XE_1AP:
    {
        /* A,B,C,D,Select,START,E1,E2 buttons -> E1(?) E2(?) START SELECT(?) A B C D */
        if(keystate[SDLK_a])  input.pad[joynum] |= INPUT_START;
        if(keystate[SDLK_s])  input.pad[joynum] |= INPUT_A;
        if(keystate[SDLK_d])  input.pad[joynum] |= INPUT_C;
        if(keystate[SDLK_f])  input.pad[joynum] |= INPUT_Y;
        if(keystate[SDLK_z])  input.pad[joynum] |= INPUT_B;
        if(keystate[SDLK_x])  input.pad[joynum] |= INPUT_X;
        if(keystate[SDLK_c])  input.pad[joynum] |= INPUT_MODE;
        if(keystate[SDLK_v])  input.pad[joynum] |= INPUT_Z;
 
        /* Left Analog Stick (bidirectional) */
        if(keystate[SDLK_UP])     input.analog[joynum][1]-=2;
        else if(keystate[SDLK_DOWN])   input.analog[joynum][1]+=2;
        else input.analog[joynum][1] = 128;
        if(keystate[SDLK_LEFT])   input.analog[joynum][0]-=2;
        else if(keystate[SDLK_RIGHT])  input.analog[joynum][0]+=2;
        else input.analog[joynum][0] = 128;
 
        /* Right Analog Stick (unidirectional) */
        if(keystate[SDLK_KP8])    input.analog[joynum+1][0]-=2;
        else if(keystate[SDLK_KP2])   input.analog[joynum+1][0]+=2;
        else if(keystate[SDLK_KP4])   input.analog[joynum+1][0]-=2;
        else if(keystate[SDLK_KP6])  input.analog[joynum+1][0]+=2;
        else input.analog[joynum+1][0] = 128;
 
        /* Limiters */
        if (input.analog[joynum][0] > 0xFF) input.analog[joynum][0] = 0xFF;
        else if (input.analog[joynum][0] < 0) input.analog[joynum][0] = 0;
        if (input.analog[joynum][1] > 0xFF) input.analog[joynum][1] = 0xFF;
        else if (input.analog[joynum][1] < 0) input.analog[joynum][1] = 0;
        if (input.analog[joynum+1][0] > 0xFF) input.analog[joynum+1][0] = 0xFF;
        else if (input.analog[joynum+1][0] < 0) input.analog[joynum+1][0] = 0;
        if (input.analog[joynum+1][1] > 0xFF) input.analog[joynum+1][1] = 0xFF;
        else if (input.analog[joynum+1][1] < 0) input.analog[joynum+1][1] = 0;
 
        break;
    }
 
    case DEVICE_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;
 
        /* Map mouse buttons to player #1 inputs */
        if(state & SDL_BUTTON_MMASK) pico_current = (pico_current + 1) & 7;
        if(state & SDL_BUTTON_RMASK) input.pad[0] |= INPUT_PICO_RED;
        if(state & SDL_BUTTON_LMASK) input.pad[0] |= INPUT_PICO_PEN;
 
        break;
    }
 
    case DEVICE_TEREBI:
    {
        /* get mouse (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
 
        /* Calculate X Y axis values */
        input.analog[0][0] = (x * 250) / VIDEO_WIDTH;
        input.analog[0][1] = (y * 250) / VIDEO_HEIGHT;
 
        /* Map mouse buttons to player #1 inputs */
        if(state & SDL_BUTTON_RMASK) input.pad[0] |= INPUT_B;
 
        break;
    }
 
    case DEVICE_GRAPHIC_BOARD:
    {
        /* get mouse (absolute values) */
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
 
        /* Calculate X Y axis values */
        input.analog[0][0] = (x * 255) / VIDEO_WIDTH;
        input.analog[0][1] = (y * 255) / VIDEO_HEIGHT;
 
        /* Map mouse buttons to player #1 inputs */
        if(state & SDL_BUTTON_LMASK) input.pad[0] |= INPUT_GRAPHIC_PEN;
        if(state & SDL_BUTTON_RMASK) input.pad[0] |= INPUT_GRAPHIC_MENU;
        if(state & SDL_BUTTON_MMASK) input.pad[0] |= INPUT_GRAPHIC_DO;
 
        break;
    }
 
    case DEVICE_ACTIVATOR:
    {
        if(keystate[SDLK_g])  input.pad[joynum] |= INPUT_ACTIVATOR_7L;
        if(keystate[SDLK_h])  input.pad[joynum] |= INPUT_ACTIVATOR_7U;
        if(keystate[SDLK_j])  input.pad[joynum] |= INPUT_ACTIVATOR_8L;
        if(keystate[SDLK_k])  input.pad[joynum] |= INPUT_ACTIVATOR_8U;
    }
#endif
    default:
    {
#ifdef GCWZERO
        if(keystate[config.buttons[A]])     input.pad[joynum] |= INPUT_A;
        if(keystate[config.buttons[B]])     input.pad[joynum] |= INPUT_B;
        if(keystate[config.buttons[C]])     input.pad[joynum] |= INPUT_C;
        if(keystate[config.buttons[START]]) input.pad[joynum] |= INPUT_START;
        if (show_lightgun == 1)
        {
            if(keystate[config.buttons[X]]) input.pad[4]    |= INPUT_A; //player 2
            if(keystate[config.buttons[Y]]) input.pad[4]    |= INPUT_B; //player 2
            if(keystate[config.buttons[Z]]) input.pad[4]    |= INPUT_C; //player 2
        } else
        if (show_lightgun == 2)
        {
            if(keystate[config.buttons[X]]) input.pad[4]    |= INPUT_A; //player 2
            if(keystate[config.buttons[Y]]) input.pad[4]    |= INPUT_B; //player 2
            if(keystate[config.buttons[Z]]) input.pad[4]    |= INPUT_C; //player 2
        } else
        {
            if(keystate[config.buttons[X]]) input.pad[joynum] |= INPUT_X;
            if(keystate[config.buttons[Y]]) input.pad[joynum] |= INPUT_Y;
            if(keystate[config.buttons[Z]]) input.pad[joynum] |= INPUT_Z;
        }
        if(keystate[config.buttons[MODE]])  input.pad[joynum] |= INPUT_MODE;
        if (keystate[SDLK_ESCAPE] && keystate[SDLK_RETURN])
        {
            gotomenu=1;
        }
        if (keystate[SDLK_ESCAPE] && keystate[SDLK_TAB])
        {
          //save to quicksave slot
            char save_state_file[256];
            sprintf(save_state_file,"%s/%s.gp1", get_save_directory(), rom_filename);
                FILE *f = fopen(save_state_file,"wb");
                if (f)
                {
                    uint8 buf[STATE_SIZE];
                    int len = state_save(buf);
                    fwrite(&buf, len, 1, f);
                    fclose(f);
                }
          //Save BMP screenshot
            char save_state_screenshot[256];
            sprintf(save_state_screenshot,"%s/%s.1.bmp", get_save_directory(), rom_filename);
            SDL_Surface* screenshot;
            if (!config.gcw0_fullscreen)
            {
                screenshot = SDL_CreateRGBSurface(SDL_HWSURFACE, sdl_video.srect.w, sdl_video.srect.h, 16, 0, 0, 0, 0);
                SDL_Rect temp;
                temp.x = 0;
                temp.y = 0;
                temp.w = sdl_video.srect.w;
                temp.h = sdl_video.srect.h;

                SDL_BlitSurface(sdl_video.surf_bitmap, &temp, screenshot, &temp);
                SDL_SaveBMP(screenshot, save_state_screenshot);
                SDL_FreeSurface(screenshot);
            }
            else
            {
                screenshot = SDL_CreateRGBSurface(SDL_HWSURFACE, gcw0_w, gcw0_h, 16, 0, 0, 0, 0);
                SDL_Rect temp;
                temp.x = 0;
                temp.y = 0;
                temp.w = gcw0_w;
                temp.h = gcw0_h;

                SDL_BlitSurface(sdl_video.surf_bitmap, &temp, screenshot, &temp);
                SDL_SaveBMP(screenshot, save_state_screenshot);
                SDL_FreeSurface(screenshot);
            }

            SDL_Delay(250);
        }
        if (keystate[SDLK_ESCAPE] && keystate[SDLK_BACKSPACE])
        {
          //load quicksave slot
            char save_state_file[256];
            sprintf(save_state_file,"%s/%s.gp1", get_save_directory(), rom_filename );
            FILE *f = fopen(save_state_file,"rb");
            if (f)
            {
                uint8 buf[STATE_SIZE];
                fread(&buf, STATE_SIZE, 1, f);
                state_load(buf);
                fclose(f);
            }
            SDL_Delay(250);

        }
#else
        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;
#endif

#ifdef GCWZERO //A-stick support
        static int MoveLeft  = 0;
        static int MoveRight = 0;
        static int MoveUp    = 0;
        static int MoveDown  = 0;
        Sint16 x_move = 0;
        Sint16 y_move = 0;
        static int lg_left   = 0;
        static int lg_right  = 0;
        static int lg_up     = 0;
        static int lg_down   = 0;
        SDL_Joystick* joy;
        if(SDL_NumJoysticks() > 0)
        {
            joy    = SDL_JoystickOpen(0);
            x_move = SDL_JoystickGetAxis(joy, 0);
            y_move = SDL_JoystickGetAxis(joy, 1);
        }

//      Control lightgun with A-stick if activated
        if (show_lightgun)
        {
            lg_left   = 0;
            lg_right  = 0;
            lg_up     = 0;
            lg_down   = 0;

            if (x_move < -1000 || x_move > 1000)
            {
                if (x_move < -1000 ) lg_left  = 1;
                if (x_move < -20000) lg_left  = 3;
                if (x_move >  1000 ) lg_right = 1;
                if (x_move >  20000) lg_right = 3;
                current_time = time(NULL); //cursor disappears after 3 seconds...
            }
            if (y_move < -1000 || y_move > 1000)
            {
                if (y_move < -1000 ) lg_up   = 1;
                if (y_move < -20000) lg_up   = 3;
                if (y_move >  1000 ) lg_down = 1;
                if (y_move >  20000) lg_down = 3;
                current_time = time(NULL);
            }
//      Keep mouse within screen, wrap around!
        int x,y;
        int state = SDL_GetMouseState(&x,&y);
        if (!config.gcw0_fullscreen)
        {
            if ((x - lg_left ) < sdl_video.drect.x ) x = VIDEO_WIDTH  - sdl_video.drect.x;
            if ((y - lg_up   ) < sdl_video.drect.y ) y = VIDEO_HEIGHT - sdl_video.drect.y;
//            if ((x + lg_right) > 288) x = 288;
//            if ((y + lg_down ) > 216) y = 216;
//            if ((x + lg_right) > 320) x = 320;
//            if ((y + lg_down ) > 240) y = 240;
            if ((x + lg_right) > VIDEO_WIDTH  - sdl_video.drect.x) x = sdl_video.drect.x;
            if ((y + lg_down ) > VIDEO_HEIGHT - sdl_video.drect.y) y = sdl_video.drect.y;

//        sdl_video.drect.x = (VIDEO_WIDTH  - sdl_video.drect.w) / 2;
  //      sdl_video.drect.y = (VIDEO_HEIGHT - sdl_video.drect.h) / 2;


        } else //scaling on
        {
            if ((x - lg_left) < 0) x = gcw0_w;
            if ((y - lg_up  ) < 0) y = gcw0_h;
            if ((x + lg_right) > gcw0_w) x = 0;
            if ((y + lg_down ) > gcw0_h) y = 0;
        }
        SDL_WarpMouse( ( x+ ( ( lg_right - lg_left ) * config.lightgun_speed ) ) ,
                       ( y+ ( ( lg_down  - lg_up    ) * config.lightgun_speed ) ) );

        } else
//      otherwise it's just mirroring the D-pad controls
        if (config.a_stick)
        {
            MoveLeft  = 0;
            MoveRight = 0;
            MoveUp    = 0;
            MoveDown  = 0;
            if (x_move < -1000 || x_move > 1000)
            {
                if (x_move < -1000) MoveLeft  = 1;
                if (x_move >  1000) MoveRight = 1;
            }
            if (y_move < -1000 || y_move > 1000)
            {
                if (y_move < -1000) MoveUp   = 1;
                if (y_move >  1000) MoveDown = 1;
            }
        }
if(show_lightgun == 1) //Genesis/MD D-pad controls player 2
{
        if(MoveUp              == 1)  input.pad[4] |= INPUT_UP;
        if(MoveDown            == 1)  input.pad[4] |= INPUT_DOWN;
        if(MoveLeft            == 1)  input.pad[4] |= INPUT_LEFT;
        if(MoveRight           == 1)  input.pad[4] |= INPUT_RIGHT;
        if(keystate[SDLK_UP]   == 1)  input.pad[joynum]      |= INPUT_UP;
        if(keystate[SDLK_DOWN] == 1)  input.pad[joynum]      |= INPUT_DOWN;
        if(keystate[SDLK_LEFT] == 1)  input.pad[joynum]      |= INPUT_LEFT;
        if(keystate[SDLK_RIGHT]== 1)  input.pad[joynum]      |= INPUT_RIGHT;
} else
if(show_lightgun == 2) //SMS D-pad controls player 2
{
        if(MoveUp              == 1)  input.pad[joynum] |= INPUT_UP;
        if(MoveDown            == 1)  input.pad[joynum] |= INPUT_DOWN;
        if(MoveLeft            == 1)  input.pad[joynum] |= INPUT_LEFT;
        if(MoveRight           == 1)  input.pad[joynum] |= INPUT_RIGHT;
        if(keystate[SDLK_UP]   == 1)  input.pad[4]      |= INPUT_UP;
        if(keystate[SDLK_DOWN] == 1)  input.pad[4]      |= INPUT_DOWN;
        if(keystate[SDLK_LEFT] == 1)  input.pad[4]      |= INPUT_LEFT;
        if(keystate[SDLK_RIGHT]== 1)  input.pad[4]      |= INPUT_RIGHT;
} else
{
        if     (keystate[SDLK_UP]    || MoveUp    == 1)  input.pad[joynum] |= INPUT_UP;
        else if(keystate[SDLK_DOWN]  || MoveDown  == 1)  input.pad[joynum] |= INPUT_DOWN;
        if     (keystate[SDLK_LEFT]  || MoveLeft  == 1)  input.pad[joynum] |= INPUT_LEFT;
        else if(keystate[SDLK_RIGHT] || MoveRight == 1)  input.pad[joynum] |= INPUT_RIGHT;
}
#else
        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;
#endif 
    }
    }
    return 1;
}
 
int main (int argc, char **argv)
{
    FILE *fp;
    int running = 1;
    atexit(shutdown);
 
    /* Print help if no game specified */
    if(argc < 2)
    {
        char caption[256];
        sprintf(caption, "Genesis Plus GX\\SDL\nusage: %s gamename\n", argv[0]);
        MessageBox(NULL, caption, "Information", 0);
        exit(1);
    }
 
    error_init();
    create_default_directories();
    /* set default config */
    set_config_defaults();
    
    /* using rom file name instead of crc code to save files */
    sprintf(rom_filename, "%s",  get_file_name(argv[1]));
 
    /* mark all BIOS as unloaded */
    system_bios = 0;
 
    /* Genesis BOOT ROM support (2KB max) */
    memset(boot_rom, 0xFF, 0x800);
    fp = fopen(MD_BIOS, "rb");
    if (fp != NULL)
    {
        int i;
 
        /* read BOOT ROM */
        fread(boot_rom, 1, 0x800, fp);
        fclose(fp);
 
        /* check BOOT ROM */
        if (!memcmp((char *)(boot_rom + 0x120),"GENESIS OS", 10))
        {
            /* mark Genesis BIOS as loaded */
            system_bios = SYSTEM_MD;
        }
 
        /* Byteswap ROM */
        for (i=0; i<0x800; i+=2)
        {
            uint8 temp = boot_rom[i];
            boot_rom[i] = boot_rom[i+1];
            boot_rom[i+1] = temp;
        }
    }
 
    /* initialize SDL */
    if(SDL_Init(0) < 0)
    {
        char caption[256];
        sprintf(caption, "SDL initialization failed");
        MessageBox(NULL, caption, "Error", 0);
        exit(1);
    }
#ifdef GCWZERO
    sdl_joystick_init();
#endif
    sdl_video_init();
    if (use_sound) sdl_sound_init();
    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;
#if defined(USE_8BPP_RENDERING)
    bitmap.pitch        = (bitmap.width * 1);
#elif defined(USE_15BPP_RENDERING)
    bitmap.pitch        = (bitmap.width * 2);
#elif defined(USE_16BPP_RENDERING)
    bitmap.pitch        = (bitmap.width * 2);
#elif defined(USE_32BPP_RENDERING)
    bitmap.pitch        = (bitmap.width * 4);
#endif
    bitmap.data         = sdl_video.surf_bitmap->pixels;
    SDL_UnlockSurface(sdl_video.surf_bitmap);
    bitmap.viewport.changed = 3;
 
    /* Load game file */
    if(!load_rom(argv[1]))
    {
        char caption[256];
        sprintf(caption, "Error loading file `%s'.", argv[1]);
        MessageBox(NULL, caption, "Error", 0);
        exit(1);
    }
 

    /* initialize system hardware */
//NOTE gcw0 second value determines framerate
    audio_init(SOUND_FREQUENCY, 0);
    system_init();
 
    /* Mega CD specific */
    char brm_file[256];
    if (system_hw == SYSTEM_MCD)
    {
        /* load internal backup RAM */
		sprintf(brm_file,"%s/", get_save_directory(), "scd.brm");
		fp = fopen(brm_file, "rb");
        if (fp!=NULL)
        {
            fread(scd.bram, 0x2000, 1, fp);
            fclose(fp);
        }
 
        /* check if internal backup RAM is formatted */
        if (memcmp(scd.bram + 0x2000 - 0x20, brm_format + 0x20, 0x20))
        {
            /* clear internal backup RAM */
            memset(scd.bram, 0x00, 0x200);
 
            /* Internal Backup RAM size fields */
            brm_format[0x10] = brm_format[0x12] = brm_format[0x14] = brm_format[0x16] = 0x00;
            brm_format[0x11] = brm_format[0x13] = brm_format[0x15] = brm_format[0x17] = (sizeof(scd.bram) / 64) - 3;
 
            /* format internal backup RAM */
            memcpy(scd.bram + 0x2000 - 0x40, brm_format, 0x40);
        }
 
        /* load cartridge backup RAM */
        if (scd.cartridge.id)
        {
			sprintf(brm_file,"%s/", get_save_directory(), "cart.brm");
			fp = fopen(brm_file, "rb");
            if (fp!=NULL)
            {
                fread(scd.cartridge.area, scd.cartridge.mask + 1, 1, fp);
                fclose(fp);
            }
 
            /* check if cartridge backup RAM is formatted */
            if (memcmp(scd.cartridge.area + scd.cartridge.mask + 1 - 0x20, brm_format + 0x20, 0x20))
            {
                /* clear cartridge backup RAM */
                memset(scd.cartridge.area, 0x00, scd.cartridge.mask + 1);
 
                /* Cartridge Backup RAM size fields */
                brm_format[0x10] = brm_format[0x12] = brm_format[0x14] = brm_format[0x16] = (((scd.cartridge.mask + 1) / 64) - 3) >> 8;
                brm_format[0x11] = brm_format[0x13] = brm_format[0x15] = brm_format[0x17] = (((scd.cartridge.mask + 1) / 64) - 3) & 0xff;
 
                /* format cartridge backup RAM */
                memcpy(scd.cartridge.area + scd.cartridge.mask + 1 - sizeof(brm_format), brm_format, sizeof(brm_format));
            }
        }
    }
 
    if (sram.on)
    {
        /* load SRAM */
        char save_file[256];
        sprintf(save_file,"%s/%s.srm", get_save_directory(), rom_filename);
        fp = fopen(save_file, "rb");
        if (fp!=NULL)
        {
            fread(sram.sram,0x10000,1, fp);
            fclose(fp);
        }
    }
 
    /* reset system hardware */
    system_reset();
 
    if(use_sound) SDL_PauseAudio(0);
 
    /* 3 frames = 50 ms (60hz) or 60 ms (50hz) */
    if(sdl_sync.sem_sync)
        SDL_SetTimer(vdp_pal ? 60 : 50, sdl_sync_timer_callback);
 
    /* emulation loop */
    while(running)
    {
        SDL_Event event;
        if (SDL_PollEvent(&event))
        {
            switch(event.type)
            {
                case SDL_USEREVENT:
                {
                    char caption[100];
                    sprintf(caption,"Genesis Plus GX - %d fps - %s)", event.user.code, (rominfo.international[0] != 0x20) ? rominfo.international : rominfo.domestic);
                    SDL_WM_SetCaption(caption, NULL);
                    break;
                }
 
                case SDL_QUIT:
                {
                    running = 0;
                    break;
                }
 
                case SDL_KEYDOWN:
                {
                    running = sdl_control_update(event.key.keysym.sym);
                    break;
                }
            }
        }
#ifdef GCWZERO
        if (do_once) 
        {
            do_once--; //don't waste write cycles!
            if (config.keepaspectratio)
            {
                FILE* aspect_ratio_file = fopen("/sys/devices/platform/jz-lcd.0/keep_aspect_ratio", "w");
                if (aspect_ratio_file)
                { 
                    fwrite("Y", 1, 1, aspect_ratio_file);
                    fclose(aspect_ratio_file);
                }
            }
            if (!config.keepaspectratio)
    	    {
                FILE* aspect_ratio_file = fopen("/sys/devices/platform/jz-lcd.0/keep_aspect_ratio", "w");
                if (aspect_ratio_file)
                { 
                    fwrite("N", 1, 1, aspect_ratio_file);
                    fclose(aspect_ratio_file);
                }
            }
	}

#endif
        sdl_video_update();
        sdl_sound_update(use_sound);

        if(!turbo_mode && sdl_sync.sem_sync && sdl_video.frames_rendered % 3 == 0)
        {
            if (!config.gcw0_frameskip || (config.gcw0_frameskip && (system_hw != SYSTEM_MCD))) //we really only need this for fmv sequences
                if(!gotomenu)
                    SDL_SemWait(sdl_sync.sem_sync);
#ifdef GCWZERO
        }
        else
        {
            if (gotomenu)
	        gcw0menu();
#endif
        }
    }

    return 0;

}