/* * Apple // emulator for *ix * * This software package is subject to the GNU General Public License * version 3 or later (your choice) as published by the Free Software * Foundation. * * Copyright 2015 Aaron Culliney * */ #include "common.h" #include "androidkeys.h" #include "json_parse_private.h" #include #include unsigned long android_deviceSampleRateHz = 0; unsigned long android_monoBufferSubmitSizeSamples = 0; unsigned long android_stereoBufferSubmitSizeSamples = 0; bool android_armArch = false; bool android_armArchV7A = false; bool android_arm64Arch = false; bool android_x86 = false; bool android_x86_64 = false; bool android_armNeonEnabled = false; bool android_x86SSSE3Enabled = false; enum { ANDROID_ACTION_DOWN = 0x0, ANDROID_ACTION_UP = 0x1, ANDROID_ACTION_MOVE = 0x2, ANDROID_ACTION_CANCEL = 0x3, ANDROID_ACTION_POINTER_DOWN = 0x5, ANDROID_ACTION_POINTER_UP = 0x6, }; typedef enum lifecycle_seq_t { APP_RUNNING = 0, APP_REQUESTED_SHUTDOWN, APP_FINISHED, } lifecycle_seq_t; static lifecycle_seq_t appState = APP_RUNNING; #if TESTING static void _start_tests(void) { char *local_argv[] = { "-f", NULL }; int local_argc = 0; for (char **p = &local_argv[0]; *p != NULL; p++) { ++local_argc; } #if TEST_CPU // Currently this test is the only one that runs as a black screen extern int test_cpu(int, char *[]); test_cpu(local_argc, local_argv); kill(getpid(), SIGKILL); // and we're done ... #endif cpu_pause(); emulator_start(); while (cpu_thread_id == 0) { sleep(1); } cpu_resume(); #if TEST_DISK extern void test_disk(int, char *[]); test_disk(local_argc, local_argv); #elif TEST_DISPLAY extern void test_display(int, char *[]); test_display(local_argc, local_argv); #elif TEST_PREFS extern void test_prefs(int, char *[]); test_prefs(local_argc, local_argv); #elif TEST_TRACE extern void test_trace(int, char *[]); test_trace(local_argc, local_argv); #elif TEST_UI extern void test_ui(int, char *[]); test_ui(local_argc, local_argv); #elif TEST_VM extern void test_vm(int, char *[]); test_vm(local_argc, local_argv); #elif TEST_CPU // handled above ... #else # error "OOPS, no tests specified" #endif } #endif static inline int _androidTouchEvent2InterfaceEvent(jint action) { switch (action) { case ANDROID_ACTION_DOWN: return TOUCH_DOWN; case ANDROID_ACTION_MOVE: return TOUCH_MOVE; case ANDROID_ACTION_UP: return TOUCH_UP; case ANDROID_ACTION_POINTER_DOWN: return TOUCH_POINTER_DOWN; case ANDROID_ACTION_POINTER_UP: return TOUCH_POINTER_UP; case ANDROID_ACTION_CANCEL: return TOUCH_CANCEL; default: LOG("Unknown Android event : %d", action); return TOUCH_CANCEL; } } static void discover_cpu_family(void) { LOG("Discovering CPU family..."); AndroidCpuFamily family = android_getCpuFamily(); uint64_t features = android_getCpuFeatures(); if (family == ANDROID_CPU_FAMILY_X86) { android_x86 = true; if (features & ANDROID_CPU_X86_FEATURE_SSSE3) { LOG("nANDROID_CPU_X86_FEATURE_SSSE3"); android_x86SSSE3Enabled = true; } if (features & ANDROID_CPU_X86_FEATURE_MOVBE) { LOG("ANDROID_CPU_X86_FEATURE_MOVBE"); } if (features & ANDROID_CPU_X86_FEATURE_POPCNT) { LOG("ANDROID_CPU_X86_FEATURE_POPCNT"); } } else if (family == ANDROID_CPU_FAMILY_ARM) { if (features & ANDROID_CPU_ARM_FEATURE_ARMv7) { LOG("ANDROID_CPU_ARM_FEATURE_ARMv7"); android_armArchV7A = true; } else { LOG("!!! NOT ANDROID_CPU_ARM_FEATURE_ARMv7"); android_armArch = true; } if (features & ANDROID_CPU_ARM_FEATURE_VFPv3) { LOG("ANDROID_CPU_ARM_FEATURE_VFPv3"); } if (features & ANDROID_CPU_ARM_FEATURE_NEON) { LOG("ANDROID_CPU_ARM_FEATURE_NEON"); android_armNeonEnabled = true; } if (features & ANDROID_CPU_ARM_FEATURE_LDREX_STREX) { LOG("ANDROID_CPU_ARM_FEATURE_LDREX_STREX"); } } else if (family == ANDROID_CPU_FAMILY_ARM64) { #warning FIXME TODO ... //android_arm64Arch = true; android_armArchV7A = true; } } // ---------------------------------------------------------------------------- // JNI functions void Java_org_deadc0de_apple2ix_Apple2Activity_nativeOnCreate(JNIEnv *env, jclass cls, jstring j_dataDir, jint sampleRate, jint monoBufferSize, jint stereoBufferSize) { const char *dataDir = (*env)->GetStringUTFChars(env, j_dataDir, 0); // Android lifecycle can call onCreate() multiple times... if (data_dir) { LOG("IGNORING multiple calls to nativeOnCreate ..."); return; } discover_cpu_family(); // Do not remove this deadc0de ... it forces a runtime load-library/link error on Gingerbread devices if we have // incorrectly compiled the app against a later version of the NDK!!! int pagesize = getpagesize(); LOG("PAGESIZE IS : %d", pagesize); data_dir = STRDUP(dataDir); if (crashHandler && crashHandler->init) { crashHandler->init(data_dir); } char *home = NULL; ASPRINTF(&home, "HOME=%s", data_dir); if (home) { putenv(home); LEAK(home); } (*env)->ReleaseStringUTFChars(env, j_dataDir, dataDir); LOG("data_dir : %s", data_dir); android_deviceSampleRateHz = (unsigned long)sampleRate; android_monoBufferSubmitSizeSamples = (unsigned long)monoBufferSize; android_stereoBufferSubmitSizeSamples = (unsigned long)stereoBufferSize; joydriver_setClampBeyondRadius(true); //#define DO_CPU65_TRACING 1 #if DO_CPU65_TRACING # warning !!!!!!!!!! this will quickly eat up disk space !!!!!!!!!! char *trfile = NULL; ASPRINTF(&trfile, "%s/%s", data_dir, "cpu_trace.txt"); cpu65_trace_begin(trfile); FREE(trfile); #endif #if TESTING _start_tests(); #else cpu_pause(); emulator_start(); #endif } void Java_org_deadc0de_apple2ix_Apple2View_nativeGraphicsInitialized(JNIEnv *env, jclass cls) { LOG("..."); _video_setRenderThread(pthread_self()); // by definition, this method is called on the render thread ... video_shutdown(); video_init(); } jboolean Java_org_deadc0de_apple2ix_Apple2Activity_nativeEmulationResume(JNIEnv *env, jclass cls) { if (!cpu_isPaused()) { return false; } LOG("..."); cpu_resume(); return true; } jboolean Java_org_deadc0de_apple2ix_Apple2Activity_nativeEmulationPause(JNIEnv *env, jclass cls) { if (appState != APP_RUNNING) { return false; } #if DO_CPU65_TRACING cpu65_trace_checkpoint(); #endif disk6_flush(0); disk6_flush(1); if (cpu_isPaused()) { return false; } LOG("..."); cpu_pause(); prefs_save(); return true; } void Java_org_deadc0de_apple2ix_Apple2View_nativeRender(JNIEnv *env, jclass cls) { SCOPE_TRACE_VIDEO("nativeRender"); if (UNLIKELY(appState != APP_RUNNING)) { if (appState == APP_REQUESTED_SHUTDOWN) { appState = APP_FINISHED; emulator_shutdown(); } return; } //#define FPS_LOG 1 #if FPS_LOG static uint32_t prevCount = 0; static uint32_t idleCount = 0; idleCount++; static struct timespec prev = { 0 }; struct timespec now; clock_gettime(CLOCK_MONOTONIC, &now); if (now.tv_sec != prev.tv_sec) { LOG("FPS : %u", idleCount-prevCount); prevCount = idleCount; prev = now; } #endif video_render(); } void Java_org_deadc0de_apple2ix_Apple2Activity_nativeReboot(JNIEnv *env, jclass cls, jint resetState) { LOG("..."); if (resetState) { // joystick button settings should be balanced by c_joystick_reset() triggered on CPU thread if (resetState == 1) { run_args.joy_button0 = 0xff; run_args.joy_button1 = 0x0; } else { run_args.joy_button0 = 0x0; run_args.joy_button1 = 0xff; } } cpu65_interrupt(ResetSig); } void Java_org_deadc0de_apple2ix_Apple2Activity_nativeOnQuit(JNIEnv *env, jclass cls) { appState = APP_REQUESTED_SHUTDOWN; LOG("..."); #if DO_CPU65_TRACING cpu65_trace_end(); #endif cpu_resume(); } void Java_org_deadc0de_apple2ix_Apple2Activity_nativeOnKeyDown(JNIEnv *env, jclass cls, jint keyCode, jint metaState) { if (UNLIKELY(appState != APP_RUNNING)) { return; } android_keycode_to_emulator(keyCode, metaState, true); } void Java_org_deadc0de_apple2ix_Apple2Activity_nativeOnKeyUp(JNIEnv *env, jclass cls, jint keyCode, jint metaState) { if (UNLIKELY(appState != APP_RUNNING)) { return; } android_keycode_to_emulator(keyCode, metaState, false); } void Java_org_deadc0de_apple2ix_Apple2View_nativeOnJoystickMove(JNIEnv *env, jclass cls, jint x, jint y) { joydriver_setAxisValue((uint8_t)x, (uint8_t)y); } jlong Java_org_deadc0de_apple2ix_Apple2View_nativeOnTouch(JNIEnv *env, jclass cls, jint action, jint pointerCount, jint pointerIndex, jfloatArray xCoords, jfloatArray yCoords) { //LOG(": %d/%d/%d :", action, pointerCount, pointerIndex); SCOPE_TRACE_INTERFACE("nativeOnTouch"); if (UNLIKELY(appState != APP_RUNNING)) { return 0x0LL; } jfloat *x_coords = (*env)->GetFloatArrayElements(env, xCoords, 0); jfloat *y_coords = (*env)->GetFloatArrayElements(env, yCoords, 0); int joyaction = _androidTouchEvent2InterfaceEvent(action); //for (unsigned int i=0; iReleaseFloatArrayElements(env, xCoords, x_coords, 0); (*env)->ReleaseFloatArrayElements(env, yCoords, y_coords, 0); return flags; } jstring Java_org_deadc0de_apple2ix_Apple2DisksMenu_nativeChooseDisk(JNIEnv *env, jclass cls, jstring jJsonString) { #if TESTING return NULL; #endif assert(cpu_isPaused() && "considered dangerous to insert disk image when CPU thread is running"); const char *jsonString = (*env)->GetStringUTFChars(env, jJsonString, NULL); JSON_ref jsonData = NULL; bool ret = json_createFromString(jsonString, &jsonData); assert(ret > 0); (*env)->ReleaseStringUTFChars(env, jJsonString, jsonString); jsonString = NULL; char *path = NULL; json_mapCopyStringValue(jsonData, "disk", &path); json_unescapeSlashes(&path); assert(path != NULL); assert(strlen(path) > 0); bool readOnly = true; json_mapParseBoolValue(jsonData, "readOnly", &readOnly); long fd = -1; if (!json_mapParseLongValue(jsonData, "fd", &fd, 10)) { TEMP_FAILURE_RETRY(fd = open(path, readOnly ? O_RDONLY : O_RDWR)); if (fd == -1) { LOG("OOPS could not open disk path : %s (%s)", path, strerror(errno)); } } else { TEMP_FAILURE_RETRY(fd = dup(fd)); if (fd == -1) { LOG("OOPS could not dup file descriptor! (%s)", strerror(errno)); } } long drive = -1; json_mapParseLongValue(jsonData, "drive", &drive, 10); assert(drive == 0 || drive == 1); bool inserted = true; const char *err = disk6_insert(fd, drive, path, readOnly); if (err) { char *diskImageUnreadable = "Disk Image Unreadable"; unsigned int cols = strlen(diskImageUnreadable); video_getAnimationDriver()->animation_showMessage(diskImageUnreadable, cols, 1); inserted = false; } else { video_getAnimationDriver()->animation_showDiskChosen(drive); } json_mapSetBoolValue(jsonData, "inserted", inserted); if (fd >= 0) { TEMP_FAILURE_RETRY(close(fd)); fd = -1; } if (path) { FREE(path); } jsonString = ((JSON_s *)jsonData)->jsonString; jstring jstr = (*env)->NewStringUTF(env, jsonString); json_destroy(&jsonData); LOG(": (fd:%d, %s, %s, %s)", (int)fd, path, drive ? "drive A" : "drive B", readOnly ? "read only" : "read/write"); return jstr; } void Java_org_deadc0de_apple2ix_Apple2DisksMenu_nativeEjectDisk(JNIEnv *env, jclass cls, jboolean driveA) { #if TESTING return; #endif LOG("..."); disk6_eject(driveA ? 0 : 1); } static int _openFdFromJson(OUTPARM int *fdOut, JSON_ref jsonData, const char * const fdKey, const char * const pathKey, int flags, int mode) { long fd = -1; char *path = NULL; do { if (!json_mapParseLongValue(jsonData, fdKey, &fd, 10)) { json_mapCopyStringValue(jsonData, pathKey, &path); assert(path != NULL); json_unescapeSlashes(&path); if (strlen(path) <= 0) { break; } if (mode == 0) { TEMP_FAILURE_RETRY(fd = open(path, flags)); } else { TEMP_FAILURE_RETRY(fd = open(path, flags, mode)); } if (fd == -1) { LOG("OOPS could not open state file path %s (%s)", path, strerror(errno)); } } else { TEMP_FAILURE_RETRY(fd = dup(fd)); if (fd == -1) { LOG("OOPS could not dup file descriptor! (%s)", strerror(errno)); } } } while (0); FREE(path); *fdOut = (int)fd; } void Java_org_deadc0de_apple2ix_Apple2Activity_nativeSaveState(JNIEnv *env, jclass cls, jstring jJsonString) { assert(cpu_isPaused() && "considered dangerous to save state when CPU thread is running"); const char *jsonString = (*env)->GetStringUTFChars(env, jJsonString, NULL); LOG(": (%s)", jsonString); JSON_ref jsonData = NULL; bool ret = json_createFromString(jsonString, &jsonData); assert(ret > 0); (*env)->ReleaseStringUTFChars(env, jJsonString, jsonString); jsonString = NULL; int fdState = -1; _openFdFromJson(&fdState, jsonData, /*fdKey:*/"fdState", /*pathKey:*/"stateFile", O_WRONLY|O_CREAT|O_TRUNC, S_IRUSR|S_IWUSR); if (!emulator_saveState(fdState)) { LOG("OOPS, could not save emulator state"); } if (fdState >= 0) { TEMP_FAILURE_RETRY(close(fdState)); fdState = -1; } json_destroy(&jsonData); } jstring Java_org_deadc0de_apple2ix_Apple2Activity_nativeLoadState(JNIEnv *env, jclass cls, jstring jJsonString) { assert(cpu_isPaused() && "considered dangerous to load state when CPU thread is running"); const char *jsonString = (*env)->GetStringUTFChars(env, jJsonString, NULL); LOG(": %s", jsonString); JSON_ref jsonData = NULL; int ret = json_createFromString(jsonString, &jsonData); assert(ret > 0); (*env)->ReleaseStringUTFChars(env, jJsonString, jsonString); jsonString = NULL; int fdState = -1; _openFdFromJson(&fdState, jsonData, /*fdKey:*/"fdState", /*pathKey:*/"stateFile", O_RDONLY, 0); int fdA = -1; { bool readOnlyA = true; json_mapParseBoolValue(jsonData, "readOnlyA", &readOnlyA); _openFdFromJson(&fdA, jsonData, /*fdKey:*/"fdA", /*pathKey:*/"diskA", readOnlyA ? O_RDONLY : O_RDWR, 0); } int fdB = -1; { bool readOnlyB = true; json_mapParseBoolValue(jsonData, "readOnlyB", &readOnlyB); _openFdFromJson(&fdB, jsonData, /*fdKey:*/"fdB", /*pathKey:*/"diskB", readOnlyB ? O_RDONLY : O_RDWR, 0); } bool loadStateSuccess = true; if (!emulator_loadState(fdState, (int)fdA, (int)fdB)) { loadStateSuccess = false; LOG("OOPS, could not load emulator state"); // FIXME TODO : should show invalid state animation here ... } if (fdState >= 0) { TEMP_FAILURE_RETRY(close(fdState)); fdState = -1; } if (fdA >= 0) { TEMP_FAILURE_RETRY(close(fdA)); fdA = -1; } if (fdB >= 0) { TEMP_FAILURE_RETRY(close(fdB)); fdB = -1; } json_mapSetBoolValue(jsonData, "loadStateSuccess", loadStateSuccess); jsonString = ((JSON_s *)jsonData)->jsonString; jstring jstr = (*env)->NewStringUTF(env, jsonString); json_destroy(&jsonData); return jstr; } jstring Java_org_deadc0de_apple2ix_Apple2Activity_nativeStateExtractDiskPaths(JNIEnv *env, jclass cls, jstring jJsonString) { assert(cpu_isPaused() && "considered dangerous to save state when CPU thread is running"); const char *jsonString = (*env)->GetStringUTFChars(env, jJsonString, NULL); LOG(": (%s)", jsonString); JSON_ref jsonData = NULL; bool ret = json_createFromString(jsonString, &jsonData); assert(ret > 0); (*env)->ReleaseStringUTFChars(env, jJsonString, jsonString); jsonString = NULL; int fdState = -1; _openFdFromJson(&fdState, jsonData, /*fdKey:*/"fdState", /*pathKey:*/"stateFile", O_RDONLY, 0); if (!emulator_stateExtractDiskPaths(fdState, jsonData)) { LOG("OOPS, could not extract disk paths from emulator state file"); } jsonString = ((JSON_s *)jsonData)->jsonString; jstring jstr = (*env)->NewStringUTF(env, jsonString); json_destroy(&jsonData); if (fdState >= 0) { TEMP_FAILURE_RETRY(close(fdState)); fdState = -1; } return jstr; } void Java_org_deadc0de_apple2ix_Apple2Preferences_nativePrefsSync(JNIEnv *env, jclass cls, jstring jDomain) { const char *domain = NULL; if (jDomain) { domain = (*env)->GetStringUTFChars(env, jDomain, 0); } #if !TEST_PREFS LOG("... domain: %s", domain); prefs_load(); prefs_sync(domain); #endif if (jDomain) { (*env)->ReleaseStringUTFChars(env, jDomain, domain); } } jlong Java_org_deadc0de_apple2ix_Apple2JoystickCalibration_nativePollJoystick(JNIEnv *env, jclass cls) { jlong cxy = 0; uint8_t c = apple_ii_64k[0][0xC000]; if (c | 0x80) { c = keys_apple2ASCII(c, NULL); } else { c = ' '; } cxy |= (c << 16); cxy |= ((joy_x & 0xFF) << 8); cxy |= ((joy_y & 0xFF) << 0); return cxy; }