2020-04-20 02:34:05 +05:30
// SPDX-License-Identifier: MPL-2.0
2020-03-28 01:06:02 +05:30
// Copyright © 2020 Skyline Team and Contributors (https://github.com/skyline-emu/)
2020-03-26 20:03:19 +05:30
# include <csignal>
2020-11-19 01:17:09 +05:30
# include <pthread.h>
2020-03-26 20:03:19 +05:30
# include <unistd.h>
2021-03-03 20:35:24 +00:00
# include <android/asset_manager_jni.h>
2021-03-03 20:42:51 +00:00
# include <sys/system_properties.h>
2019-09-25 02:24:27 +05:30
# include "skyline/common.h"
2021-09-27 18:24:07 +02:00
# include "skyline/common/language.h"
2020-12-05 23:11:52 +05:30
# include "skyline/common/signal.h"
2020-11-03 15:14:09 +05:30
# include "skyline/common/settings.h"
2021-03-20 22:22:08 +05:30
# include "skyline/common/trace.h"
2020-12-05 23:11:52 +05:30
# include "skyline/loader/loader.h"
2021-03-03 20:35:24 +00:00
# include "skyline/vfs/android_asset_filesystem.h"
2019-09-25 02:24:27 +05:30
# include "skyline/os.h"
2019-12-05 21:05:34 +05:30
# include "skyline/jvm.h"
2020-10-28 21:30:39 +05:30
# include "skyline/gpu.h"
2021-10-13 22:38:48 +05:30
# include "skyline/audio.h"
2020-04-26 05:04:35 +05:30
# include "skyline/input.h"
2020-11-03 15:14:09 +05:30
# include "skyline/kernel/types/KProcess.h"
2019-09-25 02:24:27 +05:30
2021-06-27 06:56:16 +05:30
jint Fps ; //!< An approximation of the amount of frames being submitted every second
jfloat AverageFrametimeMs ; //!< The average time it takes for a frame to be rendered and presented in milliseconds
jfloat AverageFrametimeDeviationMs ; //!< The average deviation of the average frametimes in milliseconds
2020-11-03 15:14:09 +05:30
std : : weak_ptr < skyline : : kernel : : OS > OsWeak ;
std : : weak_ptr < skyline : : gpu : : GPU > GpuWeak ;
2021-10-13 22:38:48 +05:30
std : : weak_ptr < skyline : : audio : : Audio > AudioWeak ;
2020-11-03 15:14:09 +05:30
std : : weak_ptr < skyline : : input : : Input > InputWeak ;
2019-12-02 23:10:53 +05:30
2021-03-03 20:42:51 +00:00
// https://cs.android.com/android/platform/superproject/+/master:bionic/libc/tzcode/bionic.cpp;l=43;drc=master;bpv=1;bpt=1
static std : : string GetTimeZoneName ( ) {
2021-06-27 06:56:16 +05:30
const char * nameEnv = getenv ( " TZ " ) ;
2021-03-03 20:42:51 +00:00
if ( nameEnv )
return std : : string ( nameEnv ) ;
char propBuf [ PROP_VALUE_MAX ] ;
if ( __system_property_get ( " persist.sys.timezone " , propBuf ) ) {
std : : string nameProp ( propBuf ) ;
// Flip -/+, see https://cs.android.com/android/platform/superproject/+/master:bionic/libc/tzcode/bionic.cpp;l=53;drc=master;bpv=1;bpt=1
if ( nameProp . size ( ) > 2 ) {
if ( nameProp [ 2 ] = = ' - ' )
nameProp [ 2 ] = ' + ' ;
else if ( nameProp [ 2 ] = = ' + ' )
nameProp [ 2 ] = ' - ' ;
}
return nameProp ;
}
// Fallback to GMT
return " GMT " ;
}
2021-10-27 23:27:44 +02:00
extern " C " JNIEXPORT void Java_emu_skyline_SkylineApplication_initializeLog (
JNIEnv * env ,
jobject ,
jstring appFilesPathJstring ,
jint logLevel
) {
std : : string appFilesPath { env - > GetStringUTFChars ( appFilesPathJstring , nullptr ) } ;
skyline : : Logger : : configLevel = static_cast < skyline : : Logger : : LogLevel > ( logLevel ) ;
2021-11-11 16:32:19 +01:00
skyline : : Logger : : LoaderContext . Initialize ( appFilesPath + " loader.sklog " ) ;
2021-10-27 23:27:44 +02:00
}
2021-08-28 12:30:34 +02:00
extern " C " JNIEXPORT void Java_emu_skyline_EmulationActivity_executeApplication (
2021-08-29 15:27:31 +02:00
JNIEnv * env ,
jobject instance ,
2021-08-28 12:30:34 +02:00
jstring romUriJstring ,
jint romType ,
jint romFd ,
jint preferenceFd ,
jint systemLanguage ,
jstring appFilesPathJstring ,
jobject assetManager
) {
2021-03-04 19:00:14 +05:30
skyline : : signal : : ScopedStackBlocker stackBlocker ; // We do not want anything to unwind past JNI code as there are invalid stack frames which can lead to a segmentation fault
2021-10-24 20:45:29 +01:00
Fps = 0 ;
AverageFrametimeMs = AverageFrametimeDeviationMs = 0.0f ;
2019-12-05 21:05:34 +05:30
2020-10-28 21:30:39 +05:30
pthread_setname_np ( pthread_self ( ) , " EmuMain " ) ;
2019-12-02 23:10:53 +05:30
2020-09-26 10:47:57 +05:30
auto jvmManager { std : : make_shared < skyline : : JvmManager > ( env , instance ) } ;
auto settings { std : : make_shared < skyline : : Settings > ( preferenceFd ) } ;
2020-09-29 18:16:17 +05:30
close ( preferenceFd ) ;
2020-08-08 20:38:51 +01:00
2021-11-09 21:18:47 +05:30
skyline : : JniString appFilesPath ( env , appFilesPathJstring ) ;
2021-11-11 16:32:19 +01:00
skyline : : Logger : : EmulationContext . Initialize ( appFilesPath + " emulation.sklog " ) ;
2019-12-02 23:10:53 +05:30
2020-09-26 10:47:57 +05:30
auto start { std : : chrono : : steady_clock : : now ( ) } ;
2019-12-02 23:10:53 +05:30
2021-03-12 00:11:12 +05:30
// Initialize tracing
perfetto : : TracingInitArgs args ;
args . backends | = perfetto : : kSystemBackend ;
perfetto : : Tracing : : Initialize ( args ) ;
perfetto : : TrackEvent : : Register ( ) ;
2019-09-25 02:24:27 +05:30
try {
2021-08-29 15:27:31 +02:00
auto os { std : : make_shared < skyline : : kernel : : OS > (
2021-10-13 22:38:48 +05:30
jvmManager ,
settings ,
2021-11-09 21:18:47 +05:30
appFilesPath ,
2021-10-13 22:38:48 +05:30
GetTimeZoneName ( ) ,
static_cast < skyline : : language : : SystemLanguage > ( systemLanguage ) ,
std : : make_shared < skyline : : vfs : : AndroidAssetFileSystem > ( AAssetManager_fromJava ( env , assetManager ) )
) } ;
2020-11-03 15:14:09 +05:30
OsWeak = os ;
GpuWeak = os - > state . gpu ;
2021-10-13 22:38:48 +05:30
AudioWeak = os - > state . audio ;
2020-11-03 15:14:09 +05:30
InputWeak = os - > state . input ;
2020-08-21 16:44:27 +05:30
jvmManager - > InitializeControllers ( ) ;
2020-08-08 20:38:51 +01:00
2021-11-10 23:21:43 +01:00
skyline : : Logger : : InfoNoPrefix ( " Launching ROM {} " , skyline : : JniString ( env , romUriJstring ) ) ;
2020-04-26 05:04:35 +05:30
2020-11-03 15:14:09 +05:30
os - > Execute ( romFd , static_cast < skyline : : loader : : RomFormat > ( romType ) ) ;
2019-09-25 02:24:27 +05:30
} catch ( std : : exception & e ) {
2021-11-10 23:21:43 +01:00
skyline : : Logger : : ErrorNoPrefix ( " An uncaught exception has occurred: {} " , e . what ( ) ) ;
2020-12-05 23:11:52 +05:30
} catch ( const skyline : : signal : : SignalException & e ) {
2021-11-10 23:21:43 +01:00
skyline : : Logger : : ErrorNoPrefix ( " An uncaught exception has occurred: {} " , e . what ( ) ) ;
2019-09-25 02:24:27 +05:30
} catch ( . . . ) {
2021-11-10 23:21:43 +01:00
skyline : : Logger : : ErrorNoPrefix ( " An unknown uncaught exception has occurred " ) ;
2019-09-25 02:24:27 +05:30
}
2020-04-26 05:04:35 +05:30
2021-03-12 00:11:12 +05:30
perfetto : : TrackEvent : : Flush ( ) ;
2020-11-03 15:14:09 +05:30
InputWeak . reset ( ) ;
2020-08-21 00:01:32 +05:30
2020-09-26 10:47:57 +05:30
auto end { std : : chrono : : steady_clock : : now ( ) } ;
2021-11-10 23:21:43 +01:00
skyline : : Logger : : Write ( skyline : : Logger : : LogLevel : : Info , fmt : : format ( " Emulation has ended in {}ms " , std : : chrono : : duration_cast < std : : chrono : : milliseconds > ( end - start ) . count ( ) ) ) ;
2020-09-29 18:16:17 +05:30
2021-10-27 23:27:44 +02:00
skyline : : Logger : : EmulationContext . Finalize ( ) ;
2020-09-29 18:16:17 +05:30
close ( romFd ) ;
2019-09-25 02:24:27 +05:30
}
2019-12-05 21:05:34 +05:30
2021-10-13 22:38:48 +05:30
extern " C " JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_stopEmulation ( JNIEnv * , jobject , jboolean join ) {
2020-11-03 15:14:09 +05:30
auto os { OsWeak . lock ( ) } ;
2020-11-19 01:17:09 +05:30
if ( ! os )
return false ;
2020-11-03 15:14:09 +05:30
auto process { os - > state . process } ;
2020-11-19 01:17:09 +05:30
if ( ! process )
return false ;
2021-10-13 22:38:48 +05:30
process - > Kill ( join , false , true ) ;
2020-11-19 01:17:09 +05:30
return true ;
2019-12-05 21:05:34 +05:30
}
2020-11-19 01:17:09 +05:30
extern " C " JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_setSurface ( JNIEnv * , jobject , jobject surface ) {
2020-11-03 15:14:09 +05:30
auto gpu { GpuWeak . lock ( ) } ;
2020-11-19 01:17:09 +05:30
if ( ! gpu )
return false ;
2020-10-28 21:30:39 +05:30
gpu - > presentation . UpdateSurface ( surface ) ;
2020-11-19 01:17:09 +05:30
return true ;
2019-12-05 21:05:34 +05:30
}
2020-04-18 05:46:09 +05:30
2021-10-13 22:38:48 +05:30
extern " C " JNIEXPORT void Java_emu_skyline_EmulationActivity_changeAudioStatus ( JNIEnv * , jobject , jboolean play ) {
auto audio { AudioWeak . lock ( ) } ;
if ( audio )
if ( play )
audio - > Resume ( ) ;
else
audio - > Pause ( ) ;
}
2020-11-19 01:17:09 +05:30
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 05:46:09 +05:30
2021-06-27 06:56:16 +05:30
static jfieldID averageFrametimeField { } ;
if ( ! averageFrametimeField )
averageFrametimeField = env - > GetFieldID ( clazz , " averageFrametime " , " F " ) ;
env - > SetFloatField ( thiz , averageFrametimeField , AverageFrametimeMs ) ;
static jfieldID averageFrametimeDeviationField { } ;
if ( ! averageFrametimeDeviationField )
averageFrametimeDeviationField = env - > GetFieldID ( clazz , " averageFrametimeDeviation " , " F " ) ;
env - > SetFloatField ( thiz , averageFrametimeDeviationField , AverageFrametimeDeviationMs ) ;
2020-08-08 20:38:51 +01:00
}
2020-04-26 05:04:35 +05:30
2020-08-15 19:21:23 +05:30
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setController ( JNIEnv * , jobject , jint index , jint type , jint partnerIndex ) {
2020-11-03 15:14:09 +05:30
auto input { InputWeak . lock ( ) } ;
2020-08-21 00:01:32 +05:30
std : : lock_guard guard ( input - > npad . mutex ) ;
2021-10-24 20:45:29 +01:00
input - > npad . controllers [ static_cast < size_t > ( index ) ] = skyline : : input : : GuestController { static_cast < skyline : : input : : NpadControllerType > ( type ) , static_cast < skyline : : i8 > ( partnerIndex ) } ;
2020-08-15 19:21:23 +05:30
}
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_updateControllers ( JNIEnv * , jobject ) {
2020-11-03 15:14:09 +05:30
InputWeak . lock ( ) - > npad . Update ( ) ;
2020-08-15 19:21:23 +05:30
}
2020-08-21 16:44:27 +05:30
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setButtonState ( JNIEnv * , jobject , jint index , jlong mask , jboolean pressed ) {
2020-11-03 15:14:09 +05:30
auto input { InputWeak . lock ( ) } ;
2020-10-28 21:30:39 +05:30
if ( ! input )
return ; // We don't mind if we miss button updates while input hasn't been initialized
2021-10-24 20:45:29 +01:00
auto device { input - > npad . controllers [ static_cast < size_t > ( index ) ] . device } ;
2020-10-28 21:30:39 +05:30
if ( device )
device - > SetButtonState ( skyline : : input : : NpadButton { . raw = static_cast < skyline : : u64 > ( mask ) } , pressed ) ;
2020-04-26 05:04:35 +05:30
}
2020-08-15 19:21:23 +05:30
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setAxisValue ( JNIEnv * , jobject , jint index , jint axis , jint value ) {
2020-11-03 15:14:09 +05:30
auto input { InputWeak . lock ( ) } ;
2020-10-28 21:30:39 +05:30
if ( ! input )
return ; // We don't mind if we miss axis updates while input hasn't been initialized
2021-10-24 20:45:29 +01:00
auto device { input - > npad . controllers [ static_cast < size_t > ( index ) ] . device } ;
2020-10-28 21:30:39 +05:30
if ( device )
device - > SetAxisValue ( static_cast < skyline : : input : : NpadAxisId > ( axis ) , value ) ;
2020-04-26 05:04:35 +05:30
}
2020-09-07 22:09:05 +05:30
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setTouchState ( JNIEnv * env , jobject , jintArray pointsJni ) {
2020-10-28 21:30:39 +05:30
using Point = skyline : : input : : TouchScreenPoint ;
2020-09-07 22:09:05 +05:30
2020-11-03 15:14:09 +05:30
auto input { InputWeak . lock ( ) } ;
2020-10-28 21:30:39 +05:30
if ( ! input )
return ; // We don't mind if we miss touch updates while input hasn't been initialized
jboolean isCopy { false } ;
2020-09-07 22:09:05 +05:30
2021-10-24 20:45:29 +01:00
skyline : : span < Point > points ( reinterpret_cast < Point * > ( env - > GetIntArrayElements ( pointsJni , & isCopy ) ) ,
static_cast < size_t > ( env - > GetArrayLength ( pointsJni ) ) / ( sizeof ( Point ) / sizeof ( jint ) ) ) ;
2020-10-28 21:30:39 +05:30
input - > touch . SetState ( points ) ;
env - > ReleaseIntArrayElements ( pointsJni , reinterpret_cast < jint * > ( points . data ( ) ) , JNI_ABORT ) ;
2020-09-07 22:09:05 +05:30
}