/*************************************************************************** osdepend.c OS dependant stuff (display handling, keyboard scan...) This is the only file which should me modified in order to port the emulator to a different system. ***************************************************************************/ #include #include #include #include #include #include #include "osdepend.h" #include "TwkUser.c" #define SCREEN_WIDTH 640 #define SCREEN_HEIGHT 480 #define SCREEN_MODE GFX_VESA1 int bitmap_width,bitmap_height; int first_free_pen; unsigned char *bitmap; int use_vesa; int play_sound; int noscanlines; /* put here anything you need to do when the program is started. Return 0 if */ /* initialization was successful, nonzero otherwise. */ int osd_init(int argc,char **argv) { int i; allegro_init(); install_keyboard(); /* Allegro keyboard handler */ first_free_pen = 0; use_vesa = 0; play_sound = 1; noscanlines = 0; for (i = 1;i < argc;i++) { if (stricmp(argv[i],"-vesa") == 0) use_vesa = 1; if (stricmp(argv[i],"-nosound") == 0) play_sound = 0; if (stricmp(argv[i],"-noscanlines") == 0) noscanlines = 1; } if (play_sound) { /* install digital sound driver */ return install_sound(DIGI_AUTODETECT,MIDI_NONE,0); } else return 0; } /* put here cleanup routines to be executed when the program is terminated. */ void osd_exit(void) { } Register scr224x288[] = { { 0x3c2, 0x00, 0xa7},{ 0x3d4, 0x00, 0x71},{ 0x3d4, 0x01, 0x37}, { 0x3d4, 0x02, 0x64},{ 0x3d4, 0x03, 0x92},{ 0x3d4, 0x04, 0x4f}, { 0x3d4, 0x05, 0x98},{ 0x3d4, 0x06, 0x46},{ 0x3d4, 0x07, 0x1f}, { 0x3d4, 0x08, 0x00},{ 0x3d4, 0x09, 0x40},{ 0x3d4, 0x10, 0x31}, { 0x3d4, 0x11, 0x80},{ 0x3d4, 0x12, 0x1f},{ 0x3d4, 0x13, 0x1c}, { 0x3d4, 0x14, 0x40},{ 0x3d4, 0x15, 0x2f},{ 0x3d4, 0x16, 0x44}, { 0x3d4, 0x17, 0xe3},{ 0x3c4, 0x01, 0x01},{ 0x3c4, 0x02, 0x0f}, { 0x3c4, 0x04, 0x0e},{ 0x3ce, 0x05, 0x40},{ 0x3ce, 0x06, 0x05}, { 0x3c0, 0x10, 0x41},{ 0x3c0, 0x13, 0x00} }; Register scr256x256[] = { { 0x3c2, 0x00, 0xe3},{ 0x3d4, 0x00, 0x5f},{ 0x3d4, 0x01, 0x3f}, { 0x3d4, 0x02, 0x40},{ 0x3d4, 0x03, 0x82},{ 0x3d4, 0x04, 0x4A}, { 0x3d4, 0x05, 0x9A},{ 0x3d4, 0x06, 0x23},{ 0x3d4, 0x07, 0xb2}, { 0x3d4, 0x08, 0x00},{ 0x3d4, 0x09, 0x61},{ 0x3d4, 0x10, 0x0a}, { 0x3d4, 0x11, 0xac},{ 0x3d4, 0x12, 0xff},{ 0x3d4, 0x13, 0x20}, { 0x3d4, 0x14, 0x40},{ 0x3d4, 0x15, 0x07},{ 0x3d4, 0x16, 0x1a}, { 0x3d4, 0x17, 0xa3},{ 0x3c4, 0x01, 0x01},{ 0x3c4, 0x04, 0x0e}, { 0x3ce, 0x05, 0x40},{ 0x3ce, 0x06, 0x05},{ 0x3c0, 0x10, 0x41}, { 0x3c0, 0x13, 0x00} }; Register scr256x256scanlines[] = { { 0x3c2, 0x00, 0xe3},{ 0x3d4, 0x00, 0x5f},{ 0x3d4, 0x01, 0x3f}, { 0x3d4, 0x02, 0x40},{ 0x3d4, 0x03, 0x82},{ 0x3d4, 0x04, 0x4A}, { 0x3d4, 0x05, 0x9A},{ 0x3d4, 0x06, 0x23},{ 0x3d4, 0x07, 0x1f}, { 0x3d4, 0x08, 0x00},{ 0x3d4, 0x09, 0x60},{ 0x3d4, 0x10, 0x0a}, { 0x3d4, 0x11, 0xac},{ 0x3d4, 0x12, 0x00},{ 0x3d4, 0x13, 0x20}, { 0x3d4, 0x14, 0x40},{ 0x3d4, 0x15, 0x07},{ 0x3d4, 0x16, 0x1a}, { 0x3d4, 0x17, 0xa3},{ 0x3c4, 0x01, 0x01},{ 0x3c4, 0x04, 0x0e}, { 0x3ce, 0x05, 0x40},{ 0x3ce, 0x06, 0x05},{ 0x3c0, 0x10, 0x41}, { 0x3c0, 0x13, 0x00} }; /* Create a display screen, or window, large enough to accomodate a bitmap */ /* of the given dimensions. I don't do any test here (224x288 will just do */ /* for now) but one could e.g. open a window of the exact dimensions */ /* provided. Return a bitmap pointer or 0 in case of error. */ unsigned char *osd_create_display(int width,int height) { bitmap = 0; bitmap_width = width; bitmap_height = height; if (!(width == 224 && height == 288) && !(width == 256 && height == 256)) use_vesa = 1; if (use_vesa) { if (set_gfx_mode(SCREEN_MODE,SCREEN_WIDTH,SCREEN_HEIGHT,0,0) != 0) return 0; } else { /* big hack: open a mode 13h screen using Allegro, then load the custom screen */ /* definition over it. */ if (set_gfx_mode(GFX_VGA,320,200,0,0) != 0) return 0; if (width == 224 && height == 288) outRegArray(scr224x288,sizeof(scr224x288)/sizeof(Register)); else if (width == 256 && height == 256) { if (noscanlines) outRegArray(scr256x256,sizeof(scr256x256)/sizeof(Register)); else outRegArray(scr256x256scanlines,sizeof(scr256x256)/sizeof(Register)); } } bitmap = malloc(width * height * sizeof(unsigned char)); return bitmap; } /* shut up the display */ void osd_close_display(void) { set_gfx_mode(GFX_TEXT,80,25,0,0); free(bitmap); } int osd_obtain_pen(unsigned char red, unsigned char green, unsigned char blue) { RGB rgb; rgb.r = red >> 2; rgb.g = green >> 2; rgb.b = blue >> 2; set_color(first_free_pen,&rgb); return first_free_pen++; } /* Update the display. */ /* As an additional bonus, this function also saves the screen as a PCX file */ /* when the user presses F5. This is not required for porting. */ void osd_update_display(void) { if (use_vesa) { int y; int width4 = bitmap_width / 4; unsigned long *lb = (unsigned long *)bitmap; for (y = 0;y < bitmap_height;y++) { unsigned long address; address = bmp_write_line(screen,y + (SCREEN_HEIGHT - bitmap_height) / 2) + (SCREEN_WIDTH - bitmap_width) / 2; _dosmemputl(lb,width4,address); lb += width4; } } else { /* copy the bitmap to screen memory */ _dosmemputl(bitmap,bitmap_width * bitmap_height / 4,0xa0000); } /* if the user pressed F12, save a snapshot of the screen. */ if (osd_key_pressed(OSD_KEY_F12)) { BITMAP *bmp; PALETTE pal; char name[13]; FILE *f; static int snapno; int y; do { sprintf(name,"snap%04d.pcx",snapno); /* avoid overwriting of existing files */ if ((f = fopen(name,"rb")) != 0) { fclose(f); snapno++; } } while (f != 0); get_palette(pal); bmp = create_bitmap(bitmap_width,bitmap_height); for (y = 0;y < bitmap_height;y++) memcpy(bmp->line[y],&bitmap[y * bitmap_width],bitmap_width); save_pcx(name,bmp,pal); destroy_bitmap(bmp); snapno++; /* wait for the user to release F12 */ while (osd_key_pressed(OSD_KEY_F12)); } } /* play an 8-bit sample */ void osd_play_sample(unsigned char *data,int len,int freq,int volume) { static int c; static SAMPLE sample[10]; /* allow up to 10 concurrent samples */ if (play_sound == 0) return; sample[c].bits = 8; sample[c].freq = freq; sample[c].data = data; sample[c].len = len; play_sample(&sample[c],volume,128,1000,0); c++; if (c == 10) c = 0; } /* check if a key is pressed. The keycode is the standard PC keyboard code, as */ /* defined in osdepend.h. Return 0 if the key is not pressed, nonzero otherwise. */ int osd_key_pressed(int keycode) { return key[keycode]; } /* wait for a key press and return the keycode */ int osd_read_key(void) { clear_keybuf(); return readkey() >> 8; }