/* * Copyright (c) 2014, Peter Rutenbar * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include "../core/shoebill.h" static void _print_vers(void) { printf("Shoebill v0.0.4 - http://github.com/pruten/shoebill - Peter Rutenbar (c) 2014\n\n"); } rb_tree *keymap; static void _init_keyboard_map (void) { #define mapkeymod(u, a, m) do { \ assert((a >> 7) == 0); \ uint16_t value = ((m) << 8)| (a); \ rb_insert(keymap, u, &value, NULL); \ } while (0) #define mapkey(_u, a) mapkeymod(_u, a, 0) keymap = rb_new(p_new_pool(NULL), sizeof(uint16_t)); // Letters mapkey('a', 0x00); mapkey('b', 0x0b); mapkey('c', 0x08); mapkey('d', 0x02); mapkey('e', 0x0e); mapkey('f', 0x03); mapkey('g', 0x05); mapkey('h', 0x04); mapkey('i', 0x22); mapkey('j', 0x26); mapkey('k', 0x28); mapkey('l', 0x25); mapkey('m', 0x2e); mapkey('n', 0x2d); mapkey('o', 0x1f); mapkey('p', 0x23); mapkey('q', 0x0c); mapkey('r', 0x0f); mapkey('s', 0x01); mapkey('t', 0x11); mapkey('u', 0x20); mapkey('v', 0x09); mapkey('w', 0x0d); mapkey('x', 0x07); mapkey('y', 0x10); mapkey('z', 0x06); // Numbers mapkey('0', 0x1d); mapkey('1', 0x12); mapkey('2', 0x13); mapkey('3', 0x14); mapkey('4', 0x15); mapkey('5', 0x17); mapkey('6', 0x16); mapkey('7', 0x1a); mapkey('8', 0x1c); mapkey('9', 0x19); // Top row symbols mapkeymod(')', 0x1d, modShift); mapkeymod('!', 0x12, modShift); mapkeymod('@', 0x13, modShift); mapkeymod('#', 0x14, modShift); mapkeymod('$', 0x15, modShift); mapkeymod('%', 0x17, modShift); mapkeymod('^', 0x16, modShift); mapkeymod('&', 0x1a, modShift); mapkeymod('*', 0x1c, modShift); mapkeymod('(', 0x19, modShift); // Other symbols (no shift) mapkeymod('`', 0x32, 0); mapkeymod('-', 0x1b, 0); mapkeymod('=', 0x18, 0); mapkeymod('[', 0x21, 0); mapkeymod(']', 0x1e, 0); mapkeymod('\\', 0x2a, 0); mapkeymod(';', 0x29, 0); mapkeymod('\'', 0x27, 0); mapkeymod(',', 0x2b, 0); mapkeymod('.', 0x2f, 0); mapkeymod('/', 0x2c, 0); // Other symbols (with shift) mapkeymod('~', 0x32, modShift); mapkeymod('_', 0x1b, modShift); mapkeymod('+', 0x18, modShift); mapkeymod('{', 0x21, modShift); mapkeymod('}', 0x1e, modShift); mapkeymod('|', 0x2a, modShift); mapkeymod(':', 0x29, modShift); mapkeymod('"', 0x27, modShift); mapkeymod('<', 0x2b, modShift); mapkeymod('>', 0x2f, modShift); mapkeymod('?', 0x2c, modShift); // Function keys mapkey(SDLK_F1, 0x7a); mapkey(SDLK_F2, 0x78); mapkey(SDLK_F3, 0x63); mapkey(SDLK_F4, 0x76); mapkey(SDLK_F5, 0x60); mapkey(SDLK_F6, 0x61); mapkey(SDLK_F7, 0x62); mapkey(SDLK_F8, 0x64); mapkey(SDLK_F9, 0x65); mapkey(SDLK_F10, 0x6d); mapkey(SDLK_F11, 0x67); mapkey(SDLK_F12, 0x6f); mapkey(SDLK_F13, 0x69); mapkey(SDLK_F14, 0x6b); mapkey(SDLK_F15, 0x71); // Arrows mapkey(SDLK_UP, 0x3e); mapkey(SDLK_DOWN, 0x3d); mapkey(SDLK_RIGHT, 0x3c); mapkey(SDLK_LEFT, 0x3b); // Delete mapkey(SDLK_DELETE, 0x75); mapkey(SDLK_BACKSPACE, 0x33); mapkey(SDLK_BACKSPACE, 0x33); // Enter, NL, CR mapkey(SDLK_RETURN2, 0x24); mapkey(SDLK_RETURN, 0x24); // mapkey(0x03, 0x24); // Other keys mapkey(SDLK_ESCAPE, 0x35); // escape mapkey(SDLK_SPACE, 0x31); // space mapkey(SDLK_TAB, 0x30); // tab } static void _display_frame (SDL_Window *win) { shoebill_video_frame_info_t frame = shoebill_get_video_frame(9, 0); shoebill_send_vbl_interrupt(9); glDrawBuffer(GL_BACK); glClear(GL_COLOR_BUFFER_BIT); glClearColor(0, 0, 0, 1.0); glViewport(0, 0, frame.width, frame.height); glRasterPos2i(0, frame.height); glPixelStorei(GL_UNPACK_LSB_FIRST, GL_TRUE); glPixelStorei(GL_UNPACK_ALIGNMENT, 1); glPixelZoom(1.0, -1.0); glDrawPixels(frame.width, frame.height, GL_RGBA, GL_UNSIGNED_BYTE, frame.buf); SDL_GL_SwapWindow(win); } struct shoe_app_pram_data_t { uint8_t pram[256]; FILE *f; pthread_t threadid; volatile _Bool updated, tear_down_thread; }; struct { const char *scsi_path[8]; const char *rom_path; const char *relative_unix_path; const char *pram_path; uint32_t height, width; uint32_t ram_megabytes; _Bool verbose, use_tfb; struct shoe_app_pram_data_t pram_data; } user_params; #define equals_arg(name) keylen = strlen(name); value = argv[i]+keylen; if (strncmp((name), argv[i], keylen) == 0) #if !((defined WIN32) || (defined _WIN64)) #include #include #include #endif static char* _get_home_dir (const char *terminal_element) { char *result = NULL; #if (defined WIN32) || (defined _WIN64) if (getenv("USERPROFILE") != NULL) { result = malloc(strlen(getenv("USERPROFILE")) + strlen(terminal_element) + 32); sprintf(result, "%s\\%s", getenv("USERPROFILE"), terminal_element); goto done; } if (getenv("HOMEDRIVE") && getenv("HOMEPATH")) { result = malloc(strlen(getenv("HOMEDRIVE")) + strlen(getenv("HOMEPATH")) + strlen(terminal_element) + 32); sprintf(result, "%s\\%s\\%s", getenv("HOMEDRIVE"), getenv("HOMEPATH"), terminal_element); goto done; } #else if (getenv("HOME") != NULL) { result = malloc(strlen(getenv("HOME")) + strlen(terminal_element) + 32); sprintf(result, "%s/%s", getenv("HOME"), terminal_element); goto done; } struct passwd *pwd = getpwuid(getuid()); if (pwd) { result = malloc(strlen(pwd->pw_dir) + strlen(terminal_element) + 32); sprintf(result, "%s/%s", pwd->pw_dir, terminal_element); goto done; } #endif done: // printf("_get_home_dir: debug: %s\n", result); return result; } static void _print_help (void) { printf("Arguments have the form name=value.\n"); printf("\n"); printf("rom=\n"); printf("Specifies the path to a Macintosh II ROM.\n"); printf("E.g. rom=/home/foo/macii.rom\n"); printf("\n"); printf("disk0..disk6=\n"); printf("Specifies the path to a disk image for the given SCSI ID. Shoebill will always boot from disk0, so make sure disk0 points to a bootable A/UX image.\n"); printf("E.g. disk0=/home/foo/aux3.img disk1=/blah.img\n"); printf("\n"); printf("ram=\n"); printf("E.g. ram=16\n"); printf("\n"); printf("height=\n"); printf("Specifies the height of the screen in pixels.\n"); printf("\n"); printf("width=\n"); printf("Specifies the width of the screen in pixels.\n"); printf("\n"); printf("pram-path=\n"); printf("Defaults to ~/.shoebill_pram\n"); printf("\n"); printf("verbose=<1 or 0>\n"); printf("Whether to boot A/UX in verbose mode. Best to leave it at default (1).\n"); printf("\n"); printf("unix-path=\n"); printf("Path to the kernel file on the root disk image. Best to leave it at default (/unix).\n"); printf("\n"); printf("\n"); printf("Examples:\n"); printf("\n"); printf("shoebill.exe disk0=C:\\aux3.img rom=C:\\macii.rom width=1024 height=768 ram=64\n"); printf("\n"); printf("./shoebill disk0=/aux3.img rom=/macii.rom width=1024 height=768 ram=64\n"); printf("\n"); } static void _init_user_params (int argc, char **argv) { char *key; uint32_t i; for (i=0; i<8; i++) user_params.scsi_path[i] = NULL; user_params.rom_path = "macii.rom"; user_params.relative_unix_path = "/unix"; user_params.height = 640; user_params.width = 800; user_params.ram_megabytes = 16; user_params.verbose = 1; user_params.use_tfb = 0; user_params.pram_path = _get_home_dir(".shoebill_pram"); if (argc < 2) { _print_help(); exit(0); } for (i=1; ipram[addr] = byte; pram_data->updated = 1; } void* _pram_writer_thread (void *param) { struct shoe_app_pram_data_t *pram_data = (struct shoe_app_pram_data_t*)param; while (!pram_data->tear_down_thread) { if (pram_data->updated) { pram_data->updated = 0; rewind(pram_data->f); assert(fwrite(pram_data->pram, 256, 1, pram_data->f) == 1); fflush(stdout); pram_data->tear_down_thread = 0; } sleep(1); } return NULL; } static _Bool _setup_shoebill (void) { uint32_t i; shoebill_config_t config; memset(&config, 0, sizeof(shoebill_config_t)); config.aux_verbose = user_params.verbose; config.ram_size = user_params.ram_megabytes * 1024 * 1024; config.aux_kernel_path = user_params.relative_unix_path; config.rom_path = user_params.rom_path; config.pram_callback = _pram_callback; config.pram_callback_param = (void*)&user_params.pram_data; memcpy(config.pram, user_params.pram_data.pram, 256); for (i=0; i<7; i++) config.scsi_devices[i].path = user_params.scsi_path[i]; if (!shoebill_initialize(&config)) { printf("%s\n", config.error_msg); return 0; } if (user_params.use_tfb) { shoebill_install_tfb_card(&config, 9); } else { shoebill_install_video_card(&config, 9, // slotnum user_params.width, user_params.height); } shoebill_start(); return 1; } static void _handle_key_event (SDL_Event *event) { const SDL_Keycode sym = event->key.keysym.sym; const _Bool key_down = (event->type == SDL_KEYDOWN); const SDL_Keymod sdl_mod = SDL_GetModState(); uint16_t adb_mod = 0; uint16_t value; if (sdl_mod & KMOD_SHIFT) adb_mod |= modShift; if (sdl_mod & KMOD_CTRL) adb_mod |= modControl; if (sdl_mod & KMOD_ALT) adb_mod |= modOption; if (sdl_mod & KMOD_GUI) adb_mod |= modCommand; if (sdl_mod & KMOD_CAPS) adb_mod |= modCapsLock; if (rb_find(keymap, sym, &value)) { shoebill_key_modifier((value >> 8) | adb_mod); shoebill_key(key_down, value & 0xff); } } static _Bool _init_pram (void) { FILE *f = fopen(user_params.pram_path, "r+b"); memset(&user_params.pram_data, 0, sizeof(struct shoe_app_pram_data_t)); if ((f == NULL) || (fread(user_params.pram_data.pram, 256, 1, f) != 1)) { if (f == NULL) f = fopen(user_params.pram_path, "w+b"); if (f == NULL) { printf("Can't open pram_path! [%s] [errno=%s]\n", user_params.pram_path, sys_errlist[errno]); return 0; } rewind(f); shoebill_validate_or_zap_pram(user_params.pram_data.pram, 1); assert(fwrite(user_params.pram_data.pram, 256, 1, f) == 1); fflush(f); } user_params.pram_data.f = f; shoebill_validate_or_zap_pram(user_params.pram_data.pram, 0); pthread_create(&user_params.pram_data.threadid, NULL, _pram_writer_thread, &user_params.pram_data); return 1; } int main (int argc, char **argv) { const uint32_t frame_ticks = 1000 / 60; uint32_t last_frame_ticks; _Bool capture_cursor; _print_vers(); _init_keyboard_map(); _init_user_params(argc, argv); if (!_init_pram()) return 0; else if (!_setup_shoebill()) return 0; shoebill_video_frame_info_t frame = shoebill_get_video_frame(9, 1); SDL_Init(SDL_INIT_VIDEO); SDL_Window *win = SDL_CreateWindow("Shoebill", SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, frame.width, frame.height, SDL_WINDOW_OPENGL); SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1); SDL_GLContext glctx = SDL_GL_CreateContext(win); glShadeModel(GL_FLAT); glPixelStorei(GL_UNPACK_ALIGNMENT, 1); glClearColor(0.5, 0.5, 0.5, 1.0); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); glMatrixMode(GL_PROJECTION); glLoadIdentity(); glOrtho(0, frame.width, 0, frame.height, -1.0, 1.0); capture_cursor = 1; SDL_ShowCursor(0); SDL_SetRelativeMouseMode(1); SDL_GL_SetSwapInterval(1); last_frame_ticks = SDL_GetTicks(); while (1) { const uint32_t now = SDL_GetTicks(); uint32_t ticks_to_next_frame; SDL_Event event; if ((now - last_frame_ticks) >= frame_ticks) { _display_frame(win); last_frame_ticks = now; ticks_to_next_frame = frame_ticks; } else ticks_to_next_frame = frame_ticks - (now - last_frame_ticks); event.type = SDL_USEREVENT; SDL_WaitEventTimeout(&event, ticks_to_next_frame); switch (event.type) { case SDL_QUIT: goto quit; case SDL_MOUSEBUTTONDOWN: { if ((event.button.button == SDL_BUTTON_LEFT) && capture_cursor) shoebill_mouse_click(1); break; } case SDL_MOUSEBUTTONUP: { // If the cursor isn't captured, then any click will capture it if (!capture_cursor) capture_cursor = 1; // If the cursor is captured, then left clicks get sent to the OS else if (event.button.button == SDL_BUTTON_LEFT) shoebill_mouse_click(0); // If the cursor is captured, then right clicks will uncapture it else if ((event.button.button == SDL_BUTTON_RIGHT) && capture_cursor) capture_cursor = 0; break; } case SDL_MOUSEMOTION: { if (capture_cursor) { _Bool down = event.motion.state & SDL_BUTTON(SDL_BUTTON_LEFT); shoebill_mouse_click(down); shoebill_mouse_move_delta(event.motion.xrel, event.motion.yrel); } break; } case SDL_KEYDOWN: case SDL_KEYUP: if (!event.key.repeat) _handle_key_event(&event); break; } if (capture_cursor) { SDL_ShowCursor(0); SDL_SetRelativeMouseMode(1); } else { SDL_ShowCursor(1); SDL_SetRelativeMouseMode(0); } } quit: // FIXME: tear down the pram thread and flush pram return 0; }