2020-04-19 23:04:05 +02:00
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
2020-03-27 20:36:02 +01:00
|
|
|
// Copyright © 2020 Skyline Team and Contributors (https://github.com/skyline-emu/)
|
|
|
|
|
2020-03-26 15:33:19 +01:00
|
|
|
#include <csignal>
|
2020-11-18 20:47:09 +01:00
|
|
|
#include <pthread.h>
|
2020-03-26 15:33:19 +01:00
|
|
|
#include <unistd.h>
|
2020-09-29 14:46:17 +02:00
|
|
|
#include <android/log.h>
|
2019-09-24 22:54:27 +02:00
|
|
|
#include "skyline/common.h"
|
2020-12-05 18:41:52 +01:00
|
|
|
#include "skyline/common/signal.h"
|
2020-11-03 10:44:09 +01:00
|
|
|
#include "skyline/common/settings.h"
|
2020-12-05 18:41:52 +01:00
|
|
|
#include "skyline/loader/loader.h"
|
2019-09-24 22:54:27 +02:00
|
|
|
#include "skyline/os.h"
|
2019-12-05 16:35:34 +01:00
|
|
|
#include "skyline/jvm.h"
|
2020-10-28 17:00:39 +01:00
|
|
|
#include "skyline/gpu.h"
|
2020-04-26 01:34:35 +02:00
|
|
|
#include "skyline/input.h"
|
2020-11-03 10:44:09 +01:00
|
|
|
#include "skyline/kernel/types/KProcess.h"
|
2019-09-24 22:54:27 +02:00
|
|
|
|
2020-11-03 10:44:09 +01:00
|
|
|
skyline::u16 Fps;
|
|
|
|
skyline::u32 FrameTime;
|
|
|
|
std::weak_ptr<skyline::kernel::OS> OsWeak;
|
|
|
|
std::weak_ptr<skyline::gpu::GPU> GpuWeak;
|
|
|
|
std::weak_ptr<skyline::input::Input> InputWeak;
|
2019-12-02 18:40:53 +01:00
|
|
|
|
2020-08-08 21:38:51 +02:00
|
|
|
extern "C" JNIEXPORT void Java_emu_skyline_EmulationActivity_executeApplication(JNIEnv *env, jobject instance, jstring romUriJstring, jint romType, jint romFd, jint preferenceFd, jstring appFilesPathJstring) {
|
2021-01-15 22:15:06 +01:00
|
|
|
skyline::signal::ScopedStackBlocker stackBlocker;
|
2020-11-18 20:47:09 +01:00
|
|
|
Fps = FrameTime = 0;
|
2019-12-05 16:35:34 +01:00
|
|
|
|
2020-10-28 17:00:39 +01:00
|
|
|
pthread_setname_np(pthread_self(), "EmuMain");
|
2019-12-02 18:40:53 +01:00
|
|
|
|
2020-09-26 07:17:57 +02:00
|
|
|
auto jvmManager{std::make_shared<skyline::JvmManager>(env, instance)};
|
|
|
|
auto settings{std::make_shared<skyline::Settings>(preferenceFd)};
|
2020-09-29 14:46:17 +02:00
|
|
|
close(preferenceFd);
|
2020-08-08 21:38:51 +02:00
|
|
|
|
2020-09-26 07:17:57 +02:00
|
|
|
auto appFilesPath{env->GetStringUTFChars(appFilesPathJstring, nullptr)};
|
2021-02-08 05:17:17 +01:00
|
|
|
auto logger{std::make_shared<skyline::Logger>(std::string(appFilesPath) + "skyline.log", settings->logLevel)};
|
2019-12-02 18:40:53 +01:00
|
|
|
|
2020-09-26 07:17:57 +02:00
|
|
|
auto start{std::chrono::steady_clock::now()};
|
2019-12-02 18:40:53 +01:00
|
|
|
|
2019-09-24 22:54:27 +02:00
|
|
|
try {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto os{std::make_shared<skyline::kernel::OS>(jvmManager, logger, settings, std::string(appFilesPath))};
|
|
|
|
OsWeak = os;
|
|
|
|
GpuWeak = os->state.gpu;
|
|
|
|
InputWeak = os->state.input;
|
2020-08-21 13:14:27 +02:00
|
|
|
jvmManager->InitializeControllers();
|
2020-08-08 21:38:51 +02:00
|
|
|
env->ReleaseStringUTFChars(appFilesPathJstring, appFilesPath);
|
|
|
|
|
2020-09-26 07:17:57 +02:00
|
|
|
auto romUri{env->GetStringUTFChars(romUriJstring, nullptr)};
|
2020-04-03 13:47:32 +02:00
|
|
|
logger->Info("Launching ROM {}", romUri);
|
|
|
|
env->ReleaseStringUTFChars(romUriJstring, romUri);
|
2020-04-26 01:34:35 +02:00
|
|
|
|
2020-11-03 10:44:09 +01:00
|
|
|
os->Execute(romFd, static_cast<skyline::loader::RomFormat>(romType));
|
2019-09-24 22:54:27 +02:00
|
|
|
} catch (std::exception &e) {
|
Framebuffer and NativeActivity
What was added:
* Framebuffer
* NativeActivity
* NV Services
* IOCTL Handler
* NV Devices:
* * /dev/nvmap - 0xC0080101, 0xC0080103, 0xC0200104, 0xC0180105, 0xC00C0109, 0xC008010E
* * /dev/nvhost-as-gpu
* * /dev/nvhost-channel - 0x40044801, 0xC0104809, 0xC010480B, 0xC018480C, 0x4004480D, 0xC020481A, 0x40084714
* * /dev/nvhost-ctrl
* * /dev/nvhost-ctrl-gpu - 0x80044701, 0x80284702, 0xC0184706, 0xC0B04705, 0x80084714
* SVCs:
* * SetMemoryAttribute
* * CreateTransferMemory
* * ResetSignal
* * GetSystemTick
* Addition of Compact Logger
What was fixed:
* SVCs:
* * SetHeapSize
* * SetMemoryAttribute
* * QueryMemory
* A release build would not set CMAKE_BUILD_TYPE to "RELEASE"
* The logger code was simplified
2019-11-13 21:09:31 +01:00
|
|
|
logger->Error(e.what());
|
2020-12-05 18:41:52 +01:00
|
|
|
} catch (const skyline::signal::SignalException &e) {
|
|
|
|
logger->Error(e.what());
|
2019-09-24 22:54:27 +02:00
|
|
|
} catch (...) {
|
Framebuffer and NativeActivity
What was added:
* Framebuffer
* NativeActivity
* NV Services
* IOCTL Handler
* NV Devices:
* * /dev/nvmap - 0xC0080101, 0xC0080103, 0xC0200104, 0xC0180105, 0xC00C0109, 0xC008010E
* * /dev/nvhost-as-gpu
* * /dev/nvhost-channel - 0x40044801, 0xC0104809, 0xC010480B, 0xC018480C, 0x4004480D, 0xC020481A, 0x40084714
* * /dev/nvhost-ctrl
* * /dev/nvhost-ctrl-gpu - 0x80044701, 0x80284702, 0xC0184706, 0xC0B04705, 0x80084714
* SVCs:
* * SetMemoryAttribute
* * CreateTransferMemory
* * ResetSignal
* * GetSystemTick
* Addition of Compact Logger
What was fixed:
* SVCs:
* * SetHeapSize
* * SetMemoryAttribute
* * QueryMemory
* A release build would not set CMAKE_BUILD_TYPE to "RELEASE"
* The logger code was simplified
2019-11-13 21:09:31 +01:00
|
|
|
logger->Error("An unknown exception has occurred");
|
2019-09-24 22:54:27 +02:00
|
|
|
}
|
2020-04-26 01:34:35 +02:00
|
|
|
|
2020-11-03 10:44:09 +01:00
|
|
|
InputWeak.reset();
|
2020-08-20 20:31:32 +02:00
|
|
|
|
2020-02-11 07:34:22 +01:00
|
|
|
logger->Info("Emulation has ended");
|
2019-12-02 18:40:53 +01:00
|
|
|
|
2020-09-26 07:17:57 +02:00
|
|
|
auto end{std::chrono::steady_clock::now()};
|
Framebuffer and NativeActivity
What was added:
* Framebuffer
* NativeActivity
* NV Services
* IOCTL Handler
* NV Devices:
* * /dev/nvmap - 0xC0080101, 0xC0080103, 0xC0200104, 0xC0180105, 0xC00C0109, 0xC008010E
* * /dev/nvhost-as-gpu
* * /dev/nvhost-channel - 0x40044801, 0xC0104809, 0xC010480B, 0xC018480C, 0x4004480D, 0xC020481A, 0x40084714
* * /dev/nvhost-ctrl
* * /dev/nvhost-ctrl-gpu - 0x80044701, 0x80284702, 0xC0184706, 0xC0B04705, 0x80084714
* SVCs:
* * SetMemoryAttribute
* * CreateTransferMemory
* * ResetSignal
* * GetSystemTick
* Addition of Compact Logger
What was fixed:
* SVCs:
* * SetHeapSize
* * SetMemoryAttribute
* * QueryMemory
* A release build would not set CMAKE_BUILD_TYPE to "RELEASE"
* The logger code was simplified
2019-11-13 21:09:31 +01:00
|
|
|
logger->Info("Done in: {} ms", (std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count()));
|
2020-09-29 14:46:17 +02:00
|
|
|
|
|
|
|
close(romFd);
|
2019-09-24 22:54:27 +02:00
|
|
|
}
|
2019-12-05 16:35:34 +01:00
|
|
|
|
2020-11-18 20:47:09 +01:00
|
|
|
extern "C" JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_stopEmulation(JNIEnv *, jobject) {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto os{OsWeak.lock()};
|
2020-11-18 20:47:09 +01:00
|
|
|
if (!os)
|
|
|
|
return false;
|
2020-11-03 10:44:09 +01:00
|
|
|
auto process{os->state.process};
|
2020-11-18 20:47:09 +01:00
|
|
|
if (!process)
|
|
|
|
return false;
|
2020-11-17 01:48:41 +01:00
|
|
|
process->Kill(true, false, true);
|
2020-11-18 20:47:09 +01:00
|
|
|
return true;
|
2019-12-05 16:35:34 +01:00
|
|
|
}
|
|
|
|
|
2020-11-18 20:47:09 +01:00
|
|
|
extern "C" JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_setSurface(JNIEnv *, jobject, jobject surface) {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto gpu{GpuWeak.lock()};
|
2020-11-18 20:47:09 +01:00
|
|
|
if (!gpu)
|
|
|
|
return false;
|
2020-10-28 17:00:39 +01:00
|
|
|
gpu->presentation.UpdateSurface(surface);
|
2020-11-18 20:47:09 +01:00
|
|
|
return true;
|
2019-12-05 16:35:34 +01:00
|
|
|
}
|
2020-04-18 02:16:09 +02:00
|
|
|
|
2020-11-18 20:47:09 +01:00
|
|
|
extern "C" JNIEXPORT void Java_emu_skyline_EmulationActivity_updatePerformanceStatistics(JNIEnv *env, jobject thiz) {
|
|
|
|
static jclass clazz{};
|
|
|
|
if (!clazz)
|
|
|
|
clazz = env->GetObjectClass(thiz);
|
|
|
|
|
|
|
|
static jfieldID fpsField{};
|
|
|
|
if (!fpsField)
|
|
|
|
fpsField = env->GetFieldID(clazz, "fps", "I");
|
|
|
|
env->SetIntField(thiz, fpsField, Fps);
|
2020-04-18 02:16:09 +02:00
|
|
|
|
2020-11-18 20:47:09 +01:00
|
|
|
static jfieldID frametimeField{};
|
|
|
|
if (!frametimeField)
|
|
|
|
frametimeField = env->GetFieldID(clazz, "frametime", "F");
|
|
|
|
env->SetFloatField(thiz, frametimeField, static_cast<float>(FrameTime) / 100);
|
2020-08-08 21:38:51 +02:00
|
|
|
}
|
2020-04-26 01:34:35 +02:00
|
|
|
|
2020-08-15 15:51:23 +02:00
|
|
|
extern "C" JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setController(JNIEnv *, jobject, jint index, jint type, jint partnerIndex) {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto input{InputWeak.lock()};
|
2020-08-20 20:31:32 +02:00
|
|
|
std::lock_guard guard(input->npad.mutex);
|
2020-08-15 15:51:23 +02:00
|
|
|
input->npad.controllers[index] = skyline::input::GuestController{static_cast<skyline::input::NpadControllerType>(type), static_cast<skyline::i8>(partnerIndex)};
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_updateControllers(JNIEnv *, jobject) {
|
2020-11-03 10:44:09 +01:00
|
|
|
InputWeak.lock()->npad.Update();
|
2020-08-15 15:51:23 +02:00
|
|
|
}
|
|
|
|
|
2020-08-21 13:14:27 +02:00
|
|
|
extern "C" JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setButtonState(JNIEnv *, jobject, jint index, jlong mask, jboolean pressed) {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto input{InputWeak.lock()};
|
2020-10-28 17:00:39 +01:00
|
|
|
if (!input)
|
|
|
|
return; // We don't mind if we miss button updates while input hasn't been initialized
|
|
|
|
auto device{input->npad.controllers[index].device};
|
|
|
|
if (device)
|
|
|
|
device->SetButtonState(skyline::input::NpadButton{.raw = static_cast<skyline::u64>(mask)}, pressed);
|
2020-04-26 01:34:35 +02:00
|
|
|
}
|
|
|
|
|
2020-08-15 15:51:23 +02:00
|
|
|
extern "C" JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setAxisValue(JNIEnv *, jobject, jint index, jint axis, jint value) {
|
2020-11-03 10:44:09 +01:00
|
|
|
auto input{InputWeak.lock()};
|
2020-10-28 17:00:39 +01:00
|
|
|
if (!input)
|
|
|
|
return; // We don't mind if we miss axis updates while input hasn't been initialized
|
|
|
|
auto device{input->npad.controllers[index].device};
|
|
|
|
if (device)
|
|
|
|
device->SetAxisValue(static_cast<skyline::input::NpadAxisId>(axis), value);
|
2020-04-26 01:34:35 +02:00
|
|
|
}
|
2020-09-07 18:39:05 +02:00
|
|
|
|
|
|
|
extern "C" JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setTouchState(JNIEnv *env, jobject, jintArray pointsJni) {
|
2020-10-28 17:00:39 +01:00
|
|
|
using Point = skyline::input::TouchScreenPoint;
|
2020-09-07 18:39:05 +02:00
|
|
|
|
2020-11-03 10:44:09 +01:00
|
|
|
auto input{InputWeak.lock()};
|
2020-10-28 17:00:39 +01:00
|
|
|
if (!input)
|
|
|
|
return; // We don't mind if we miss touch updates while input hasn't been initialized
|
|
|
|
jboolean isCopy{false};
|
2020-09-07 18:39:05 +02:00
|
|
|
|
2020-10-28 17:00:39 +01:00
|
|
|
skyline::span<Point> points(reinterpret_cast<Point *>(env->GetIntArrayElements(pointsJni, &isCopy)), env->GetArrayLength(pointsJni) / (sizeof(Point) / sizeof(jint)));
|
|
|
|
input->touch.SetState(points);
|
|
|
|
env->ReleaseIntArrayElements(pointsJni, reinterpret_cast<jint *>(points.data()), JNI_ABORT);
|
2020-09-07 18:39:05 +02:00
|
|
|
}
|