2020-04-20 00:04:05 +03:00
// SPDX-License-Identifier: MPL-2.0
2020-03-27 22:36:02 +03:00
// Copyright © 2020 Skyline Team and Contributors (https://github.com/skyline-emu/)
2020-03-26 17:33:19 +03:00
# include <csignal>
2020-11-18 22:47:09 +03:00
# include <pthread.h>
2020-03-26 17:33:19 +03:00
# include <unistd.h>
2020-09-29 15:46:17 +03:00
# include <android/log.h>
2021-03-03 23:35:24 +03:00
# include <android/asset_manager_jni.h>
2021-03-03 23:42:51 +03:00
# include <sys/system_properties.h>
2019-09-24 23:54:27 +03:00
# include "skyline/common.h"
2021-09-27 19:24:07 +03:00
# include "skyline/common/language.h"
2020-12-05 20:41:52 +03:00
# include "skyline/common/signal.h"
2020-11-03 12:44:09 +03:00
# include "skyline/common/settings.h"
2021-03-20 19:52:08 +03:00
# include "skyline/common/trace.h"
2020-12-05 20:41:52 +03:00
# include "skyline/loader/loader.h"
2021-03-03 23:35:24 +03:00
# include "skyline/vfs/android_asset_filesystem.h"
2019-09-24 23:54:27 +03:00
# include "skyline/os.h"
2019-12-05 18:35:34 +03:00
# include "skyline/jvm.h"
2020-10-28 19:00:39 +03:00
# include "skyline/gpu.h"
2021-10-13 20:08:48 +03:00
# include "skyline/audio.h"
2020-04-26 02:34:35 +03:00
# include "skyline/input.h"
2020-11-03 12:44:09 +03:00
# include "skyline/kernel/types/KProcess.h"
2019-09-24 23:54:27 +03:00
2021-06-27 04:26:16 +03:00
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 12:44:09 +03:00
std : : weak_ptr < skyline : : kernel : : OS > OsWeak ;
std : : weak_ptr < skyline : : gpu : : GPU > GpuWeak ;
2021-10-13 20:08:48 +03:00
std : : weak_ptr < skyline : : audio : : Audio > AudioWeak ;
2020-11-03 12:44:09 +03:00
std : : weak_ptr < skyline : : input : : Input > InputWeak ;
2019-12-02 20:40:53 +03:00
2021-03-03 23:42:51 +03: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 04:26:16 +03:00
const char * nameEnv = getenv ( " TZ " ) ;
2021-03-03 23:42:51 +03: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-08-28 13:30:34 +03:00
extern " C " JNIEXPORT void Java_emu_skyline_EmulationActivity_executeApplication (
2021-08-29 16:27:31 +03:00
JNIEnv * env ,
jobject instance ,
2021-08-28 13:30:34 +03:00
jstring romUriJstring ,
jint romType ,
jint romFd ,
jint preferenceFd ,
jint systemLanguage ,
jstring appFilesPathJstring ,
jobject assetManager
) {
2021-03-04 16:30:14 +03:00
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 22:45:29 +03:00
Fps = 0 ;
AverageFrametimeMs = AverageFrametimeDeviationMs = 0.0f ;
2019-12-05 18:35:34 +03:00
2020-10-28 19:00:39 +03:00
pthread_setname_np ( pthread_self ( ) , " EmuMain " ) ;
2019-12-02 20:40:53 +03:00
2020-09-26 08:17:57 +03:00
auto jvmManager { std : : make_shared < skyline : : JvmManager > ( env , instance ) } ;
auto settings { std : : make_shared < skyline : : Settings > ( preferenceFd ) } ;
2020-09-29 15:46:17 +03:00
close ( preferenceFd ) ;
2020-08-08 22:38:51 +03:00
2021-11-09 18:48:47 +03:00
skyline : : JniString appFilesPath ( env , appFilesPathJstring ) ;
auto logger { std : : make_shared < skyline : : Logger > ( appFilesPath + " skyline.log " , settings - > logLevel ) } ;
2019-12-02 20:40:53 +03:00
2020-09-26 08:17:57 +03:00
auto start { std : : chrono : : steady_clock : : now ( ) } ;
2019-12-02 20:40:53 +03:00
2021-03-11 21:41:12 +03:00
// Initialize tracing
perfetto : : TracingInitArgs args ;
args . backends | = perfetto : : kSystemBackend ;
perfetto : : Tracing : : Initialize ( args ) ;
perfetto : : TrackEvent : : Register ( ) ;
2019-09-24 23:54:27 +03:00
try {
2021-08-29 16:27:31 +03:00
auto os { std : : make_shared < skyline : : kernel : : OS > (
2021-10-13 20:08:48 +03:00
jvmManager ,
logger ,
settings ,
2021-11-09 18:48:47 +03:00
appFilesPath ,
2021-10-13 20:08:48 +03:00
GetTimeZoneName ( ) ,
static_cast < skyline : : language : : SystemLanguage > ( systemLanguage ) ,
std : : make_shared < skyline : : vfs : : AndroidAssetFileSystem > ( AAssetManager_fromJava ( env , assetManager ) )
) } ;
2020-11-03 12:44:09 +03:00
OsWeak = os ;
GpuWeak = os - > state . gpu ;
2021-10-13 20:08:48 +03:00
AudioWeak = os - > state . audio ;
2020-11-03 12:44:09 +03:00
InputWeak = os - > state . input ;
2020-08-21 14:14:27 +03:00
jvmManager - > InitializeControllers ( ) ;
2020-08-08 22:38:51 +03:00
2021-11-09 18:48:47 +03:00
logger - > InfoNoPrefix ( " Launching ROM {} " , skyline : : JniString ( env , romUriJstring ) ) ;
2020-04-26 02:34:35 +03:00
2020-11-03 12:44:09 +03:00
os - > Execute ( romFd , static_cast < skyline : : loader : : RomFormat > ( romType ) ) ;
2019-09-24 23:54:27 +03:00
} catch ( std : : exception & e ) {
2021-03-28 00:17:35 +03:00
logger - > ErrorNoPrefix ( " An uncaught exception has occurred: {} " , e . what ( ) ) ;
2020-12-05 20:41:52 +03:00
} catch ( const skyline : : signal : : SignalException & e ) {
2021-03-28 00:17:35 +03:00
logger - > ErrorNoPrefix ( " An uncaught exception has occurred: {} " , e . what ( ) ) ;
2019-09-24 23:54:27 +03:00
} catch ( . . . ) {
2021-03-28 00:17:35 +03:00
logger - > ErrorNoPrefix ( " An unknown uncaught exception has occurred " ) ;
2019-09-24 23:54:27 +03:00
}
2020-04-26 02:34:35 +03:00
2021-03-11 21:41:12 +03:00
perfetto : : TrackEvent : : Flush ( ) ;
2020-11-03 12:44:09 +03:00
InputWeak . reset ( ) ;
2020-08-20 21:31:32 +03:00
2020-09-26 08:17:57 +03:00
auto end { std : : chrono : : steady_clock : : now ( ) } ;
2021-10-14 17:18:24 +03:00
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 15:46:17 +03:00
close ( romFd ) ;
2019-09-24 23:54:27 +03:00
}
2019-12-05 18:35:34 +03:00
2021-10-13 20:08:48 +03:00
extern " C " JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_stopEmulation ( JNIEnv * , jobject , jboolean join ) {
2020-11-03 12:44:09 +03:00
auto os { OsWeak . lock ( ) } ;
2020-11-18 22:47:09 +03:00
if ( ! os )
return false ;
2020-11-03 12:44:09 +03:00
auto process { os - > state . process } ;
2020-11-18 22:47:09 +03:00
if ( ! process )
return false ;
2021-10-13 20:08:48 +03:00
process - > Kill ( join , false , true ) ;
2020-11-18 22:47:09 +03:00
return true ;
2019-12-05 18:35:34 +03:00
}
2020-11-18 22:47:09 +03:00
extern " C " JNIEXPORT jboolean Java_emu_skyline_EmulationActivity_setSurface ( JNIEnv * , jobject , jobject surface ) {
2020-11-03 12:44:09 +03:00
auto gpu { GpuWeak . lock ( ) } ;
2020-11-18 22:47:09 +03:00
if ( ! gpu )
return false ;
2020-10-28 19:00:39 +03:00
gpu - > presentation . UpdateSurface ( surface ) ;
2020-11-18 22:47:09 +03:00
return true ;
2019-12-05 18:35:34 +03:00
}
2020-04-18 03:16:09 +03:00
2021-10-13 20:08:48 +03:00
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-18 22:47:09 +03: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 03:16:09 +03:00
2021-06-27 04:26:16 +03:00
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 22:38:51 +03:00
}
2020-04-26 02:34:35 +03:00
2020-08-15 16:51:23 +03:00
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setController ( JNIEnv * , jobject , jint index , jint type , jint partnerIndex ) {
2020-11-03 12:44:09 +03:00
auto input { InputWeak . lock ( ) } ;
2020-08-20 21:31:32 +03:00
std : : lock_guard guard ( input - > npad . mutex ) ;
2021-10-24 22:45:29 +03: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 16:51:23 +03:00
}
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_updateControllers ( JNIEnv * , jobject ) {
2020-11-03 12:44:09 +03:00
InputWeak . lock ( ) - > npad . Update ( ) ;
2020-08-15 16:51:23 +03:00
}
2020-08-21 14:14:27 +03:00
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setButtonState ( JNIEnv * , jobject , jint index , jlong mask , jboolean pressed ) {
2020-11-03 12:44:09 +03:00
auto input { InputWeak . lock ( ) } ;
2020-10-28 19:00:39 +03:00
if ( ! input )
return ; // We don't mind if we miss button updates while input hasn't been initialized
2021-10-24 22:45:29 +03:00
auto device { input - > npad . controllers [ static_cast < size_t > ( index ) ] . device } ;
2020-10-28 19:00:39 +03:00
if ( device )
device - > SetButtonState ( skyline : : input : : NpadButton { . raw = static_cast < skyline : : u64 > ( mask ) } , pressed ) ;
2020-04-26 02:34:35 +03:00
}
2020-08-15 16:51:23 +03:00
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setAxisValue ( JNIEnv * , jobject , jint index , jint axis , jint value ) {
2020-11-03 12:44:09 +03:00
auto input { InputWeak . lock ( ) } ;
2020-10-28 19:00:39 +03:00
if ( ! input )
return ; // We don't mind if we miss axis updates while input hasn't been initialized
2021-10-24 22:45:29 +03:00
auto device { input - > npad . controllers [ static_cast < size_t > ( index ) ] . device } ;
2020-10-28 19:00:39 +03:00
if ( device )
device - > SetAxisValue ( static_cast < skyline : : input : : NpadAxisId > ( axis ) , value ) ;
2020-04-26 02:34:35 +03:00
}
2020-09-07 19:39:05 +03:00
extern " C " JNIEXPORT void JNICALL Java_emu_skyline_EmulationActivity_setTouchState ( JNIEnv * env , jobject , jintArray pointsJni ) {
2020-10-28 19:00:39 +03:00
using Point = skyline : : input : : TouchScreenPoint ;
2020-09-07 19:39:05 +03:00
2020-11-03 12:44:09 +03:00
auto input { InputWeak . lock ( ) } ;
2020-10-28 19:00:39 +03: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 19:39:05 +03:00
2021-10-24 22:45:29 +03: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 19:00:39 +03:00
input - > touch . SetState ( points ) ;
env - > ReleaseIntArrayElements ( pointsJni , reinterpret_cast < jint * > ( points . data ( ) ) , JNI_ABORT ) ;
2020-09-07 19:39:05 +03:00
}