2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2012-11-04 09:56:22 +00:00
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2012-11-01 15:19:01 +00:00
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
|
|
|
// NativeApp implementation for platforms that will use that framework, like:
|
|
|
|
// Android, Linux, MacOSX.
|
|
|
|
//
|
|
|
|
// Native is a cross platform framework. It's not very mature and mostly
|
|
|
|
// just built according to the needs of my own apps.
|
|
|
|
//
|
|
|
|
// Windows has its own code that bypasses the framework entirely.
|
|
|
|
|
2017-02-24 19:50:27 +00:00
|
|
|
#include "ppsspp_config.h"
|
2013-04-13 19:24:07 +00:00
|
|
|
|
|
|
|
// Background worker threads should be spawned in NativeInit and joined
|
|
|
|
// in NativeShutdown.
|
|
|
|
|
2013-04-25 08:41:43 +00:00
|
|
|
#include <locale.h>
|
2014-04-26 06:32:43 +00:00
|
|
|
#include <algorithm>
|
2014-11-25 19:58:19 +00:00
|
|
|
#include <memory>
|
2017-02-27 20:57:46 +00:00
|
|
|
#include <mutex>
|
2018-09-01 20:57:20 +00:00
|
|
|
#include <thread>
|
2014-04-26 06:32:43 +00:00
|
|
|
|
2014-06-18 05:36:49 +00:00
|
|
|
#if defined(_WIN32)
|
2014-06-22 12:17:57 +00:00
|
|
|
#include "Windows/DSoundStream.h"
|
2015-09-19 11:14:05 +00:00
|
|
|
#include "Windows/MainWindow.h"
|
2013-12-07 17:06:15 +00:00
|
|
|
#endif
|
2013-04-13 19:24:07 +00:00
|
|
|
|
2014-02-10 11:38:23 +00:00
|
|
|
#include "base/display.h"
|
2017-11-21 11:22:38 +00:00
|
|
|
#include "base/timeutil.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
#include "base/logging.h"
|
|
|
|
#include "base/NativeApp.h"
|
|
|
|
#include "file/vfs.h"
|
|
|
|
#include "file/zip_read.h"
|
2013-11-26 13:04:29 +00:00
|
|
|
#include "net/http_client.h"
|
2017-03-06 09:51:28 +00:00
|
|
|
#include "net/resolve.h"
|
2013-08-30 12:47:28 +00:00
|
|
|
#include "gfx_es2/draw_text.h"
|
2015-09-06 10:23:14 +00:00
|
|
|
#include "gfx_es2/gpu_features.h"
|
2013-04-18 09:58:54 +00:00
|
|
|
#include "i18n/i18n.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
#include "input/input_state.h"
|
2014-03-22 08:26:28 +00:00
|
|
|
#include "math/fast/fast_math.h"
|
2013-01-10 22:49:33 +00:00
|
|
|
#include "math/math_util.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
#include "math/lin/matrix4x4.h"
|
2015-05-15 15:14:54 +00:00
|
|
|
#include "profiler/profiler.h"
|
2014-08-17 10:19:04 +00:00
|
|
|
#include "thin3d/thin3d.h"
|
2013-10-17 14:15:04 +00:00
|
|
|
#include "ui/ui.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
#include "ui/screen.h"
|
2013-03-30 18:23:20 +00:00
|
|
|
#include "ui/ui_context.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
#include "ui/view.h"
|
2013-11-26 13:04:29 +00:00
|
|
|
#include "util/text/utf8.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
|
2014-03-22 08:26:28 +00:00
|
|
|
#include "Common/CPUDetect.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
#include "Common/FileUtil.h"
|
|
|
|
#include "Common/LogManager.h"
|
2014-12-20 16:31:56 +00:00
|
|
|
#include "Common/MemArena.h"
|
2015-12-31 16:25:48 +00:00
|
|
|
#include "Common/GraphicsContext.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
#include "Core/Config.h"
|
2018-06-17 01:42:31 +00:00
|
|
|
#include "Core/ConfigValues.h"
|
2014-06-22 12:17:57 +00:00
|
|
|
#include "Core/Core.h"
|
2015-06-28 01:32:21 +00:00
|
|
|
#include "Core/FileLoaders/DiskCachingFileLoader.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
#include "Core/Host.h"
|
2016-09-29 05:35:09 +00:00
|
|
|
#include "Core/Reporting.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
#include "Core/SaveState.h"
|
2014-12-21 07:14:46 +00:00
|
|
|
#include "Core/Screenshot.h"
|
2013-12-29 23:11:29 +00:00
|
|
|
#include "Core/System.h"
|
2015-01-11 11:02:49 +00:00
|
|
|
#include "Core/HLE/__sceAudio.h"
|
2013-12-29 23:11:29 +00:00
|
|
|
#include "Core/HLE/sceCtrl.h"
|
2017-08-20 18:03:06 +00:00
|
|
|
#include "Core/HLE/sceUsbCam.h"
|
|
|
|
#include "Core/HLE/sceUsbGps.h"
|
2013-11-20 13:42:48 +00:00
|
|
|
#include "Core/Util/GameManager.h"
|
2015-01-11 21:50:52 +00:00
|
|
|
#include "Core/Util/AudioFormat.h"
|
2018-04-21 20:51:18 +00:00
|
|
|
#include "Core/WebServer.h"
|
2016-09-29 05:35:09 +00:00
|
|
|
#include "GPU/GPUInterface.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
#include "ui_atlas.h"
|
2016-09-29 05:35:09 +00:00
|
|
|
#include "UI/EmuScreen.h"
|
|
|
|
#include "UI/GameInfoCache.h"
|
|
|
|
#include "UI/HostTypes.h"
|
2013-06-22 20:27:59 +00:00
|
|
|
#include "UI/OnScreenDisplay.h"
|
2013-10-17 14:15:04 +00:00
|
|
|
#include "UI/MiscScreens.h"
|
2017-12-11 02:30:28 +00:00
|
|
|
#include "UI/RemoteISOScreen.h"
|
2013-11-11 11:52:04 +00:00
|
|
|
#include "UI/TiltEventProcessor.h"
|
2014-06-22 15:02:04 +00:00
|
|
|
#include "UI/BackgroundAudio.h"
|
2016-12-27 21:26:49 +00:00
|
|
|
#include "UI/TextureUtil.h"
|
2018-08-12 22:08:56 +00:00
|
|
|
#include "UI/DiscordIntegration.h"
|
2013-06-01 21:34:50 +00:00
|
|
|
|
2014-04-26 05:54:32 +00:00
|
|
|
#if !defined(MOBILE_DEVICE)
|
|
|
|
#include "Common/KeyMap.h"
|
|
|
|
#endif
|
|
|
|
|
2018-03-03 07:46:12 +00:00
|
|
|
#if !defined(MOBILE_DEVICE) && defined(USING_QT_UI)
|
|
|
|
#include "Qt/QtHost.h"
|
|
|
|
#endif
|
2018-06-09 18:34:12 +00:00
|
|
|
#if defined(USING_QT_UI)
|
|
|
|
#include <QFontDatabase>
|
|
|
|
#endif
|
2018-03-03 07:46:12 +00:00
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
// The new UI framework, for initialization
|
|
|
|
|
|
|
|
static UI::Theme ui_theme;
|
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#if defined(ARM) && defined(__ANDROID__)
|
2013-03-30 20:14:18 +00:00
|
|
|
#include "../../android/jni/ArmEmitterTest.h"
|
2016-10-12 09:13:16 +00:00
|
|
|
#elif defined(ARM64) && defined(__ANDROID__)
|
2015-03-21 18:22:24 +00:00
|
|
|
#include "../../android/jni/Arm64EmitterTest.h"
|
2013-03-21 19:52:33 +00:00
|
|
|
#endif
|
2012-11-23 18:41:35 +00:00
|
|
|
|
2013-06-25 21:18:16 +00:00
|
|
|
#ifdef IOS
|
|
|
|
#include "ios/iOSCoreAudio.h"
|
2013-12-07 05:06:12 +00:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
#include <mach-o/dyld.h>
|
2013-06-25 21:18:16 +00:00
|
|
|
#endif
|
|
|
|
|
2013-11-03 00:13:17 +00:00
|
|
|
// https://github.com/richq/android-ndk-profiler
|
|
|
|
#ifdef ANDROID_NDK_PROFILER
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "android/android-ndk-profiler/prof.h"
|
|
|
|
#endif
|
|
|
|
|
2012-11-04 10:54:45 +00:00
|
|
|
ScreenManager *screenManager;
|
2012-11-01 15:19:01 +00:00
|
|
|
std::string config_filename;
|
|
|
|
|
2017-11-10 12:13:56 +00:00
|
|
|
bool g_graphicsInited;
|
|
|
|
|
2013-06-22 20:27:59 +00:00
|
|
|
// Really need to clean this mess of globals up... but instead I add more :P
|
|
|
|
bool g_TakeScreenshot;
|
2013-10-31 10:06:54 +00:00
|
|
|
static bool isOuya;
|
2014-12-25 16:41:12 +00:00
|
|
|
static bool resized = false;
|
2017-04-15 23:30:37 +00:00
|
|
|
static bool restarting = false;
|
2014-02-13 16:47:02 +00:00
|
|
|
|
2018-03-24 11:51:54 +00:00
|
|
|
static bool askedForStoragePermission = false;
|
2018-09-07 01:31:09 +00:00
|
|
|
static int renderCounter = 0;
|
2018-03-24 11:51:54 +00:00
|
|
|
|
2014-02-13 16:47:02 +00:00
|
|
|
struct PendingMessage {
|
|
|
|
std::string msg;
|
|
|
|
std::string value;
|
|
|
|
};
|
|
|
|
|
2017-02-27 20:57:46 +00:00
|
|
|
static std::mutex pendingMutex;
|
2014-02-13 16:47:02 +00:00
|
|
|
static std::vector<PendingMessage> pendingMessages;
|
2017-01-30 13:33:38 +00:00
|
|
|
static Draw::DrawContext *g_draw;
|
2016-12-26 12:42:53 +00:00
|
|
|
static Draw::Pipeline *colorPipeline;
|
|
|
|
static Draw::Pipeline *texColorPipeline;
|
2013-03-30 18:23:20 +00:00
|
|
|
static UIContext *uiContext;
|
2016-02-21 03:23:00 +00:00
|
|
|
static std::vector<std::string> inputboxValue;
|
2013-03-29 19:51:14 +00:00
|
|
|
|
2015-01-11 20:00:56 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
WindowsAudioBackend *winAudioBackend;
|
|
|
|
#endif
|
|
|
|
|
2013-04-13 19:24:07 +00:00
|
|
|
std::thread *graphicsLoadThread;
|
2013-03-29 19:51:14 +00:00
|
|
|
|
2014-02-04 11:09:14 +00:00
|
|
|
class AndroidLogger : public LogListener {
|
2012-11-01 15:19:01 +00:00
|
|
|
public:
|
2017-03-18 09:47:10 +00:00
|
|
|
void Log(const LogMessage &message) override {
|
|
|
|
// Log with simplified headers as Android already provides timestamp etc.
|
|
|
|
switch (message.level) {
|
2013-03-11 05:31:47 +00:00
|
|
|
case LogTypes::LVERBOSE:
|
2012-11-04 09:56:22 +00:00
|
|
|
case LogTypes::LDEBUG:
|
|
|
|
case LogTypes::LINFO:
|
2017-03-18 09:47:10 +00:00
|
|
|
ILOG("[%s] %s", message.log, message.msg.c_str());
|
2012-11-04 09:56:22 +00:00
|
|
|
break;
|
|
|
|
case LogTypes::LERROR:
|
2017-03-18 09:47:10 +00:00
|
|
|
ELOG("[%s] %s", message.log, message.msg.c_str());
|
2012-11-04 09:56:22 +00:00
|
|
|
break;
|
|
|
|
case LogTypes::LWARNING:
|
2017-03-18 09:47:10 +00:00
|
|
|
WLOG("[%s] %s", message.log, message.msg.c_str());
|
2012-11-04 09:56:22 +00:00
|
|
|
break;
|
|
|
|
case LogTypes::LNOTICE:
|
|
|
|
default:
|
2017-03-18 09:47:10 +00:00
|
|
|
ILOG("[%s] !!! %s", message.log, message.msg.c_str());
|
2012-11-04 09:56:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
};
|
|
|
|
|
2014-06-22 12:17:57 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
int Win32Mix(short *buffer, int numSamples, int bits, int rate, int channels) {
|
|
|
|
return NativeMix(buffer, numSamples);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
// globals
|
2017-12-31 08:12:34 +00:00
|
|
|
static AndroidLogger *logger = nullptr;
|
2012-11-01 15:19:01 +00:00
|
|
|
std::string boot_filename = "";
|
|
|
|
|
2015-01-11 11:02:49 +00:00
|
|
|
void NativeHost::InitSound() {
|
2013-06-25 21:18:16 +00:00
|
|
|
#ifdef IOS
|
2013-11-29 15:33:17 +00:00
|
|
|
iOSCoreAudioInit();
|
2013-06-25 21:18:16 +00:00
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
void NativeHost::ShutdownSound() {
|
2013-06-25 21:18:16 +00:00
|
|
|
#ifdef IOS
|
2013-11-29 15:33:17 +00:00
|
|
|
iOSCoreAudioShutdown();
|
2013-06-25 21:18:16 +00:00
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2014-03-01 15:09:16 +00:00
|
|
|
#if !defined(MOBILE_DEVICE) && defined(USING_QT_UI)
|
2015-01-11 11:02:49 +00:00
|
|
|
void QtHost::InitSound() { }
|
|
|
|
void QtHost::ShutdownSound() { }
|
2014-03-01 15:09:16 +00:00
|
|
|
#endif
|
|
|
|
|
2014-02-04 11:09:14 +00:00
|
|
|
std::string NativeQueryConfig(std::string query) {
|
2014-07-18 19:49:54 +00:00
|
|
|
char temp[128];
|
2014-02-04 11:58:37 +00:00
|
|
|
if (query == "screenRotation") {
|
2015-12-13 21:25:58 +00:00
|
|
|
ILOG("g_Config.screenRotation = %d", g_Config.iScreenRotation);
|
|
|
|
snprintf(temp, sizeof(temp), "%d", g_Config.iScreenRotation);
|
2014-07-18 19:49:54 +00:00
|
|
|
return std::string(temp);
|
2014-02-09 22:16:08 +00:00
|
|
|
} else if (query == "immersiveMode") {
|
2014-07-18 19:49:54 +00:00
|
|
|
return std::string(g_Config.bImmersiveMode ? "1" : "0");
|
|
|
|
} else if (query == "hwScale") {
|
2014-07-18 22:42:22 +00:00
|
|
|
int scale = g_Config.iAndroidHwScale;
|
|
|
|
if (scale == 1) {
|
|
|
|
// If g_Config.iInternalResolution is also set to Auto (1), we fall back to "Device resolution" (0). It works out.
|
|
|
|
scale = g_Config.iInternalResolution;
|
|
|
|
} else if (scale >= 2) {
|
|
|
|
scale -= 1;
|
|
|
|
}
|
|
|
|
|
2015-01-29 23:28:53 +00:00
|
|
|
int max_res = std::max(System_GetPropertyInt(SYSPROP_DISPLAY_XRES), System_GetPropertyInt(SYSPROP_DISPLAY_YRES)) / 480 + 1;
|
2015-12-13 21:25:58 +00:00
|
|
|
snprintf(temp, sizeof(temp), "%d", std::min(scale, max_res));
|
2014-07-18 19:49:54 +00:00
|
|
|
return std::string(temp);
|
2015-01-11 16:28:46 +00:00
|
|
|
} else if (query == "force44khz") {
|
|
|
|
return std::string("0");
|
2016-07-01 18:06:06 +00:00
|
|
|
} else if (query == "androidJavaGL") {
|
|
|
|
// If we're using Vulkan, we say no... need C++ to use Vulkan.
|
|
|
|
if (GetGPUBackend() == GPUBackend::VULKAN) {
|
|
|
|
return "false";
|
|
|
|
}
|
|
|
|
// Otherwise, some devices prefer the Java init so play it safe.
|
|
|
|
return "true";
|
2017-08-16 08:23:27 +00:00
|
|
|
} else if (query == "sustainedPerformanceMode") {
|
|
|
|
return std::string(g_Config.bSustainedPerformanceMode ? "1" : "0");
|
2014-02-04 11:58:37 +00:00
|
|
|
} else {
|
2015-01-11 16:28:46 +00:00
|
|
|
return "";
|
2014-02-04 11:58:37 +00:00
|
|
|
}
|
2014-02-04 11:09:14 +00:00
|
|
|
}
|
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
int NativeMix(short *audio, int num_samples) {
|
2015-01-29 13:58:03 +00:00
|
|
|
if (GetUIState() != UISTATE_INGAME) {
|
|
|
|
PlayBackgroundAudio();
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
2014-06-22 12:17:57 +00:00
|
|
|
|
2015-01-29 13:58:03 +00:00
|
|
|
int sample_rate = System_GetPropertyInt(SYSPROP_AUDIO_SAMPLE_RATE);
|
|
|
|
num_samples = __AudioMix(audio, num_samples, sample_rate > 0 ? sample_rate : 44100);
|
|
|
|
|
2014-06-22 12:17:57 +00:00
|
|
|
#ifdef _WIN32
|
2017-08-08 14:32:56 +00:00
|
|
|
winAudioBackend->Update();
|
2014-06-22 12:17:57 +00:00
|
|
|
#endif
|
|
|
|
|
2013-06-04 21:54:37 +00:00
|
|
|
return num_samples;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2014-02-04 11:09:14 +00:00
|
|
|
// This is called before NativeInit so we do a little bit of initialization here.
|
2014-12-18 21:43:54 +00:00
|
|
|
void NativeGetAppInfo(std::string *app_dir_name, std::string *app_nice_name, bool *landscape, std::string *version) {
|
2012-11-01 15:19:01 +00:00
|
|
|
*app_nice_name = "PPSSPP";
|
|
|
|
*app_dir_name = "ppsspp";
|
|
|
|
*landscape = true;
|
2014-12-18 21:43:54 +00:00
|
|
|
*version = PPSSPP_GIT_VERSION;
|
2012-11-23 18:41:35 +00:00
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#if defined(ARM) && defined(__ANDROID__)
|
2013-02-11 22:10:11 +00:00
|
|
|
ArmEmitterTest();
|
2016-10-12 09:13:16 +00:00
|
|
|
#elif defined(ARM64) && defined(__ANDROID__)
|
2015-03-21 18:22:24 +00:00
|
|
|
Arm64EmitterTest();
|
2013-03-02 00:36:18 +00:00
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2018-06-09 18:34:12 +00:00
|
|
|
#if defined(USING_WIN_UI) && !PPSSPP_PLATFORM(UWP)
|
2018-04-29 18:45:32 +00:00
|
|
|
static bool CheckFontIsUsable(const wchar_t *fontFace) {
|
2015-10-04 22:11:36 +00:00
|
|
|
wchar_t actualFontFace[1024] = { 0 };
|
|
|
|
|
|
|
|
HFONT f = CreateFont(0, 0, 0, 0, FW_LIGHT, 0, FALSE, FALSE, DEFAULT_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, PROOF_QUALITY, VARIABLE_PITCH, fontFace);
|
|
|
|
if (f != nullptr) {
|
|
|
|
HDC hdc = CreateCompatibleDC(nullptr);
|
|
|
|
if (hdc != nullptr) {
|
|
|
|
SelectObject(hdc, f);
|
|
|
|
GetTextFace(hdc, 1024, actualFontFace);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
}
|
|
|
|
DeleteObject(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we were able to get the font name, did it load?
|
|
|
|
if (actualFontFace[0] != 0) {
|
|
|
|
return wcsncmp(actualFontFace, fontFace, ARRAY_SIZE(actualFontFace)) == 0;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-04-29 18:45:32 +00:00
|
|
|
static void PostLoadConfig() {
|
|
|
|
// On Windows, we deal with currentDirectory in InitSysDirectories().
|
|
|
|
#ifndef _WIN32
|
|
|
|
if (g_Config.currentDirectory.empty()) {
|
|
|
|
#if defined(__ANDROID__)
|
|
|
|
g_Config.currentDirectory = g_Config.externalDirectory;
|
|
|
|
#elif defined(IOS)
|
|
|
|
g_Config.currentDirectory = g_Config.internalDataDirectory;
|
|
|
|
#else
|
|
|
|
if (getenv("HOME") != nullptr)
|
|
|
|
g_Config.currentDirectory = getenv("HOME");
|
|
|
|
else
|
|
|
|
g_Config.currentDirectory = "./";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Allow the lang directory to be overridden for testing purposes (e.g. Android, where it's hard to
|
|
|
|
// test new languages without recompiling the entire app, which is a hassle).
|
2018-09-02 17:27:11 +00:00
|
|
|
const std::string langOverridePath = GetSysDirectory(DIRECTORY_SYSTEM) + "lang/";
|
2018-04-29 18:45:32 +00:00
|
|
|
|
|
|
|
// If we run into the unlikely case that "lang" is actually a file, just use the built-in translations.
|
|
|
|
if (!File::Exists(langOverridePath) || !File::IsDirectory(langOverridePath))
|
|
|
|
i18nrepo.LoadIni(g_Config.sLanguageIni);
|
|
|
|
else
|
|
|
|
i18nrepo.LoadIni(g_Config.sLanguageIni, langOverridePath);
|
|
|
|
}
|
|
|
|
|
2018-05-10 17:32:28 +00:00
|
|
|
void CreateDirectoriesAndroid() {
|
|
|
|
// On Android, create a PSP directory tree in the external_dir,
|
|
|
|
// to hopefully reduce confusion a bit.
|
|
|
|
ILOG("Creating %s", (g_Config.memStickDirectory + "PSP").c_str());
|
2018-09-02 17:27:11 +00:00
|
|
|
File::CreateFullPath(g_Config.memStickDirectory + "PSP");
|
|
|
|
File::CreateFullPath(GetSysDirectory(DIRECTORY_SAVEDATA));
|
|
|
|
File::CreateFullPath(GetSysDirectory(DIRECTORY_SAVESTATE));
|
|
|
|
File::CreateFullPath(GetSysDirectory(DIRECTORY_GAME));
|
|
|
|
File::CreateFullPath(GetSysDirectory(DIRECTORY_SYSTEM));
|
2018-05-10 18:52:21 +00:00
|
|
|
|
2018-05-10 17:32:28 +00:00
|
|
|
// Avoid media scanners in PPSSPP_STATE and SAVEDATA directories
|
2018-09-02 17:27:11 +00:00
|
|
|
File::CreateEmptyFile(GetSysDirectory(DIRECTORY_SAVESTATE) + ".nomedia");
|
|
|
|
File::CreateEmptyFile(GetSysDirectory(DIRECTORY_SAVEDATA) + ".nomedia");
|
2018-09-02 17:19:08 +00:00
|
|
|
File::CreateEmptyFile(GetSysDirectory(DIRECTORY_SYSTEM) + ".nomedia");
|
2018-05-10 17:32:28 +00:00
|
|
|
}
|
|
|
|
|
2018-09-01 20:57:20 +00:00
|
|
|
static void CheckFailedGPUBackends() {
|
2018-09-06 14:31:15 +00:00
|
|
|
// We only want to do this once per process run and backend, to detect process crashes.
|
|
|
|
// If NativeShutdown is called before we finish, we might call this multiple times.
|
|
|
|
static int lastBackend = -1;
|
|
|
|
if (lastBackend == g_Config.iGPUBackend) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
lastBackend = g_Config.iGPUBackend;
|
|
|
|
|
2018-09-01 20:57:20 +00:00
|
|
|
std::string cache = GetSysDirectory(DIRECTORY_APP_CACHE) + "/FailedGraphicsBackends.txt";
|
|
|
|
|
|
|
|
if (System_GetPropertyBool(SYSPROP_SUPPORTS_PERMISSIONS)) {
|
|
|
|
std::string data;
|
|
|
|
if (readFileToString(true, cache.c_str(), data))
|
|
|
|
g_Config.sFailedGPUBackends = data;
|
|
|
|
}
|
|
|
|
|
2018-09-01 21:16:39 +00:00
|
|
|
// Use this if you want to debug a graphics crash...
|
|
|
|
if (g_Config.sFailedGPUBackends == "IGNORE")
|
|
|
|
return;
|
2018-09-06 06:13:15 +00:00
|
|
|
else if (!g_Config.sFailedGPUBackends.empty())
|
|
|
|
ERROR_LOG(LOADER, "Failed graphics backends: %s", g_Config.sFailedGPUBackends.c_str());
|
2018-09-01 21:16:39 +00:00
|
|
|
|
2018-09-01 20:57:20 +00:00
|
|
|
// Okay, let's not try a backend in the failed list.
|
2018-09-06 14:31:15 +00:00
|
|
|
g_Config.iGPUBackend = g_Config.NextValidBackend();
|
|
|
|
if (lastBackend != g_Config.iGPUBackend)
|
|
|
|
WARN_LOG(LOADER, "Failed graphics backend switched from %d to %d", lastBackend, g_Config.iGPUBackend);
|
2018-09-01 20:57:20 +00:00
|
|
|
// And then let's - for now - add the current to the failed list.
|
|
|
|
if (g_Config.sFailedGPUBackends.empty()) {
|
|
|
|
g_Config.sFailedGPUBackends = StringFromFormat("%d", g_Config.iGPUBackend);
|
|
|
|
} else if (g_Config.sFailedGPUBackends.find("ALL") == std::string::npos) {
|
|
|
|
g_Config.sFailedGPUBackends += StringFromFormat(",%d", g_Config.iGPUBackend);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (System_GetPropertyBool(SYSPROP_SUPPORTS_PERMISSIONS)) {
|
|
|
|
// Let's try to create, in case it doesn't exist.
|
|
|
|
if (!File::Exists(GetSysDirectory(DIRECTORY_APP_CACHE)))
|
|
|
|
File::CreateDir(GetSysDirectory(DIRECTORY_APP_CACHE));
|
|
|
|
writeStringToFile(true, g_Config.sFailedGPUBackends, cache.c_str());
|
|
|
|
} else {
|
|
|
|
// Just save immediately, since we have storage.
|
|
|
|
g_Config.Save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ClearFailedGPUBackends() {
|
2018-09-01 21:16:39 +00:00
|
|
|
if (g_Config.sFailedGPUBackends == "IGNORE")
|
|
|
|
return;
|
|
|
|
|
2018-09-01 20:57:20 +00:00
|
|
|
// We've successfully started graphics without crashing, hurray.
|
|
|
|
// In case they update drivers and have totally different problems much later, clear the failed list.
|
|
|
|
g_Config.sFailedGPUBackends.clear();
|
|
|
|
if (System_GetPropertyBool(SYSPROP_SUPPORTS_PERMISSIONS)) {
|
|
|
|
File::Delete(GetSysDirectory(DIRECTORY_APP_CACHE) + "/FailedGraphicsBackends.txt");
|
|
|
|
} else {
|
|
|
|
g_Config.Save();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-09 23:28:15 +00:00
|
|
|
void NativeInit(int argc, const char *argv[], const char *savegame_dir, const char *external_dir, const char *cache_dir) {
|
2017-03-12 16:24:46 +00:00
|
|
|
net::Init(); // This needs to happen before we load the config. So on Windows we also run it in Main. It's fine to call multiple times.
|
2017-03-06 09:51:28 +00:00
|
|
|
|
2014-03-22 08:26:28 +00:00
|
|
|
InitFastMath(cpu_info.bNEON);
|
2015-01-11 21:50:52 +00:00
|
|
|
SetupAudioFormats();
|
2014-03-22 08:26:28 +00:00
|
|
|
|
2018-08-12 22:08:56 +00:00
|
|
|
g_Discord.SetPresenceMenu();
|
|
|
|
|
2017-08-08 13:58:25 +00:00
|
|
|
// Make sure UI state is MENU.
|
|
|
|
ResetUIState();
|
|
|
|
|
2013-07-27 11:07:34 +00:00
|
|
|
bool skipLogo = false;
|
2013-04-25 08:41:43 +00:00
|
|
|
setlocale( LC_ALL, "C" );
|
2016-01-17 21:11:28 +00:00
|
|
|
std::string user_data_path = savegame_dir;
|
2014-02-13 16:47:02 +00:00
|
|
|
pendingMessages.clear();
|
2013-07-09 21:27:25 +00:00
|
|
|
#ifdef IOS
|
2013-03-16 06:52:52 +00:00
|
|
|
user_data_path += "/";
|
2013-07-09 21:27:25 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// We want this to be FIRST.
|
2018-06-09 17:35:40 +00:00
|
|
|
#if defined(IOS)
|
2013-07-09 21:27:25 +00:00
|
|
|
// Packed assets are included in app
|
2016-01-17 21:53:06 +00:00
|
|
|
VFSRegister("", new DirectoryAssetReader(external_dir));
|
2017-03-19 15:21:54 +00:00
|
|
|
#endif
|
|
|
|
#if !defined(MOBILE_DEVICE) && !defined(_WIN32)
|
2013-12-07 05:06:12 +00:00
|
|
|
VFSRegister("", new DirectoryAssetReader((File::GetExeDirectory() + "assets/").c_str()));
|
|
|
|
VFSRegister("", new DirectoryAssetReader((File::GetExeDirectory()).c_str()));
|
2014-08-05 09:04:34 +00:00
|
|
|
VFSRegister("", new DirectoryAssetReader("/usr/share/ppsspp/assets/"));
|
2013-04-11 05:14:44 +00:00
|
|
|
#endif
|
2016-10-16 12:47:47 +00:00
|
|
|
VFSRegister("", new DirectoryAssetReader("assets/"));
|
2016-01-17 21:11:28 +00:00
|
|
|
VFSRegister("", new DirectoryAssetReader(savegame_dir));
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2017-05-05 13:53:48 +00:00
|
|
|
#if (defined(MOBILE_DEVICE) || !defined(USING_QT_UI)) && !PPSSPP_PLATFORM(UWP)
|
2017-04-29 19:02:07 +00:00
|
|
|
if (host == nullptr) {
|
|
|
|
host = new NativeHost();
|
|
|
|
}
|
2014-03-01 15:09:16 +00:00
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2016-01-17 21:11:28 +00:00
|
|
|
g_Config.internalDataDirectory = savegame_dir;
|
2018-04-29 18:45:32 +00:00
|
|
|
g_Config.externalDirectory = external_dir;
|
|
|
|
|
|
|
|
#if defined(__ANDROID__)
|
2013-10-12 23:02:03 +00:00
|
|
|
// Maybe there should be an option to use internal memory instead, but I think
|
|
|
|
// that for most people, using external memory (SDCard/USB Storage) makes the
|
|
|
|
// most sense.
|
2016-01-17 21:11:28 +00:00
|
|
|
g_Config.memStickDirectory = std::string(external_dir) + "/";
|
|
|
|
g_Config.flash0Directory = std::string(external_dir) + "/flash0/";
|
2016-10-12 10:32:20 +00:00
|
|
|
#elif defined(IOS)
|
2014-10-19 21:20:51 +00:00
|
|
|
g_Config.memStickDirectory = user_data_path;
|
2016-01-17 21:11:28 +00:00
|
|
|
g_Config.flash0Directory = std::string(external_dir) + "/flash0/";
|
2013-10-12 23:02:03 +00:00
|
|
|
#elif !defined(_WIN32)
|
2014-06-25 06:10:00 +00:00
|
|
|
std::string config;
|
|
|
|
if (getenv("XDG_CONFIG_HOME") != NULL)
|
|
|
|
config = getenv("XDG_CONFIG_HOME");
|
|
|
|
else if (getenv("HOME") != NULL)
|
|
|
|
config = getenv("HOME") + std::string("/.config");
|
|
|
|
else // Just in case
|
|
|
|
config = "./config";
|
|
|
|
|
2014-10-19 21:20:51 +00:00
|
|
|
g_Config.memStickDirectory = config + "/ppsspp/";
|
2017-03-31 21:50:17 +00:00
|
|
|
g_Config.flash0Directory = File::GetExeDirectory() + "/assets/flash0/";
|
2013-10-12 23:02:03 +00:00
|
|
|
#endif
|
|
|
|
|
2016-02-27 21:10:08 +00:00
|
|
|
if (cache_dir && strlen(cache_dir)) {
|
|
|
|
DiskCachingFileLoaderCache::SetCacheDir(cache_dir);
|
|
|
|
g_Config.appCacheDirectory = cache_dir;
|
|
|
|
}
|
|
|
|
|
2017-04-15 23:30:37 +00:00
|
|
|
if (!LogManager::GetInstance())
|
|
|
|
LogManager::Init();
|
|
|
|
|
2013-03-29 19:03:11 +00:00
|
|
|
#ifndef _WIN32
|
2013-10-12 23:02:03 +00:00
|
|
|
g_Config.AddSearchPath(user_data_path);
|
2018-09-02 17:27:11 +00:00
|
|
|
g_Config.AddSearchPath(GetSysDirectory(DIRECTORY_SYSTEM));
|
|
|
|
g_Config.SetDefaultPath(GetSysDirectory(DIRECTORY_SYSTEM));
|
2018-03-24 11:51:54 +00:00
|
|
|
|
|
|
|
// Note that if we don't have storage permission here, loading the config will
|
|
|
|
// fail and it will be set to the default. Later, we load again when we get permission.
|
2013-10-12 23:02:03 +00:00
|
|
|
g_Config.Load();
|
2013-04-01 01:22:27 +00:00
|
|
|
#endif
|
2016-01-10 12:08:19 +00:00
|
|
|
LogManager *logman = LogManager::GetInstance();
|
2012-12-25 07:34:19 +00:00
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#ifdef __ANDROID__
|
2018-05-10 17:32:28 +00:00
|
|
|
CreateDirectoriesAndroid();
|
2013-06-19 11:13:23 +00:00
|
|
|
#endif
|
|
|
|
|
2012-12-26 23:18:45 +00:00
|
|
|
const char *fileToLog = 0;
|
2013-04-01 01:16:29 +00:00
|
|
|
const char *stateToLoad = 0;
|
2012-12-26 23:18:45 +00:00
|
|
|
|
2017-11-21 11:22:38 +00:00
|
|
|
bool gotBootFilename = false;
|
|
|
|
|
2012-11-04 10:31:06 +00:00
|
|
|
// Parse command line
|
|
|
|
LogTypes::LOG_LEVELS logLevel = LogTypes::LINFO;
|
|
|
|
for (int i = 1; i < argc; i++) {
|
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
switch (argv[i][1]) {
|
|
|
|
case 'd':
|
|
|
|
// Enable debug logging
|
2013-01-08 23:12:02 +00:00
|
|
|
// Note that you must also change the max log level in Log.h.
|
2012-11-04 10:31:06 +00:00
|
|
|
logLevel = LogTypes::LDEBUG;
|
|
|
|
break;
|
2016-09-10 15:16:53 +00:00
|
|
|
case 'v':
|
|
|
|
// Enable verbose logging
|
|
|
|
// Note that you must also change the max log level in Log.h.
|
|
|
|
logLevel = LogTypes::LVERBOSE;
|
2012-12-01 22:20:08 +00:00
|
|
|
break;
|
2012-12-05 06:55:24 +00:00
|
|
|
case 'j':
|
2017-03-02 11:36:54 +00:00
|
|
|
g_Config.iCpuCore = (int)CPUCore::JIT;
|
2012-12-25 07:34:19 +00:00
|
|
|
g_Config.bSaveSettings = false;
|
2012-12-22 17:49:59 +00:00
|
|
|
break;
|
2012-12-05 06:55:24 +00:00
|
|
|
case 'i':
|
2017-03-02 11:36:54 +00:00
|
|
|
g_Config.iCpuCore = (int)CPUCore::INTERPRETER;
|
2016-05-07 23:43:27 +00:00
|
|
|
g_Config.bSaveSettings = false;
|
|
|
|
break;
|
|
|
|
case 'r':
|
2017-03-02 11:36:54 +00:00
|
|
|
g_Config.iCpuCore = (int)CPUCore::IR_JIT;
|
2012-12-25 07:34:19 +00:00
|
|
|
g_Config.bSaveSettings = false;
|
2012-12-05 06:55:24 +00:00
|
|
|
break;
|
2012-12-26 23:18:45 +00:00
|
|
|
case '-':
|
|
|
|
if (!strncmp(argv[i], "--log=", strlen("--log=")) && strlen(argv[i]) > strlen("--log="))
|
|
|
|
fileToLog = argv[i] + strlen("--log=");
|
2013-04-01 07:28:40 +00:00
|
|
|
if (!strncmp(argv[i], "--state=", strlen("--state=")) && strlen(argv[i]) > strlen("--state="))
|
|
|
|
stateToLoad = argv[i] + strlen("--state=");
|
2016-09-03 05:15:25 +00:00
|
|
|
if (!strncmp(argv[1], "--PS3", strlen("--PS3")))
|
|
|
|
g_Config.bPS3Controller = true;
|
2014-04-26 05:00:45 +00:00
|
|
|
#if !defined(MOBILE_DEVICE)
|
2014-04-26 05:14:08 +00:00
|
|
|
if (!strncmp(argv[i], "--escape-exit", strlen("--escape-exit")))
|
2014-04-26 05:54:32 +00:00
|
|
|
g_Config.bPauseExitsEmulator = true;
|
2014-04-26 05:00:45 +00:00
|
|
|
#endif
|
2018-04-02 14:31:26 +00:00
|
|
|
if (!strncmp(argv[i], "--pause-menu-exit", strlen("--pause-menu-exit")))
|
|
|
|
g_Config.bPauseMenuExitsEmulator = true;
|
2018-06-09 23:28:15 +00:00
|
|
|
if (!strcmp(argv[i], "--fullscreen"))
|
|
|
|
g_Config.bFullScreen = true;
|
2012-12-26 23:18:45 +00:00
|
|
|
break;
|
2012-11-04 10:31:06 +00:00
|
|
|
}
|
|
|
|
} else {
|
2017-11-21 11:22:38 +00:00
|
|
|
// This parameter should be a boot filename. Only accept it if we
|
|
|
|
// don't already have one.
|
|
|
|
if (!gotBootFilename) {
|
|
|
|
gotBootFilename = true;
|
|
|
|
ILOG("Boot filename found in args: '%s'", argv[i]);
|
|
|
|
|
|
|
|
bool okToLoad = true;
|
2018-04-02 06:28:36 +00:00
|
|
|
bool okToCheck = true;
|
2017-11-21 11:22:38 +00:00
|
|
|
if (System_GetPropertyBool(SYSPROP_SUPPORTS_PERMISSIONS)) {
|
|
|
|
PermissionStatus status = System_GetPermissionStatus(SYSTEM_PERMISSION_STORAGE);
|
2018-04-02 06:28:36 +00:00
|
|
|
if (status == PERMISSION_STATUS_DENIED) {
|
|
|
|
ELOG("Storage permission denied. Launching without argument.");
|
2017-11-21 11:22:38 +00:00
|
|
|
okToLoad = false;
|
2018-04-02 06:28:36 +00:00
|
|
|
okToCheck = false;
|
|
|
|
} else if (status != PERMISSION_STATUS_GRANTED) {
|
|
|
|
ELOG("Storage permission not granted. Launching without argument check.");
|
|
|
|
okToCheck = false;
|
2017-11-21 11:22:38 +00:00
|
|
|
} else {
|
|
|
|
ILOG("Storage permission granted.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (okToLoad) {
|
|
|
|
boot_filename = argv[i];
|
2017-02-23 08:30:40 +00:00
|
|
|
#ifdef _WIN32
|
2017-11-21 11:22:38 +00:00
|
|
|
boot_filename = ReplaceAll(boot_filename, "\\", "/");
|
2017-02-23 08:30:40 +00:00
|
|
|
#endif
|
2017-11-21 11:22:38 +00:00
|
|
|
skipLogo = true;
|
2018-04-02 06:28:36 +00:00
|
|
|
}
|
|
|
|
if (okToLoad && okToCheck) {
|
2017-11-21 11:22:38 +00:00
|
|
|
std::unique_ptr<FileLoader> fileLoader(ConstructFileLoader(boot_filename));
|
|
|
|
if (!fileLoader->Exists()) {
|
|
|
|
fprintf(stderr, "File not found: %s\n", boot_filename.c_str());
|
2018-05-09 22:59:21 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
// Ignore and proceed.
|
|
|
|
#else
|
|
|
|
// Bail.
|
2017-11-21 11:22:38 +00:00
|
|
|
exit(1);
|
2018-05-09 22:59:21 +00:00
|
|
|
#endif
|
2017-11-21 11:22:38 +00:00
|
|
|
}
|
2012-12-01 22:38:18 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "Can only boot one file");
|
2018-05-09 22:59:21 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
// Ignore and proceed.
|
|
|
|
#else
|
|
|
|
// Bail.
|
2012-11-04 10:31:06 +00:00
|
|
|
exit(1);
|
2018-05-09 22:59:21 +00:00
|
|
|
#endif
|
2012-11-04 10:31:06 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-16 13:59:37 +00:00
|
|
|
if (fileToLog)
|
2012-12-26 23:18:45 +00:00
|
|
|
LogManager::GetInstance()->ChangeFileLog(fileToLog);
|
|
|
|
|
2018-04-29 18:45:32 +00:00
|
|
|
PostLoadConfig();
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2017-03-06 10:44:35 +00:00
|
|
|
// Hard reset the logs. TODO: Get rid of this and read from config.
|
|
|
|
#ifndef _WIN32
|
2016-09-10 10:24:20 +00:00
|
|
|
for (int i = 0; i < LogTypes::NUMBER_OF_LOGS; i++) {
|
2012-11-04 09:56:22 +00:00
|
|
|
LogTypes::LOG_TYPE type = (LogTypes::LOG_TYPE)i;
|
2017-03-18 09:47:10 +00:00
|
|
|
logman->SetEnabled(type, true);
|
2016-09-10 15:16:53 +00:00
|
|
|
logman->SetLogLevel(type, logLevel);
|
2017-03-18 10:11:27 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
#endif
|
2017-12-31 08:12:34 +00:00
|
|
|
|
|
|
|
#if defined(__ANDROID__) || (defined(MOBILE_DEVICE) && !defined(_DEBUG))
|
|
|
|
// Enable basic logging for any kind of mobile device, since LogManager doesn't.
|
|
|
|
// The MOBILE_DEVICE/_DEBUG condition matches LogManager.cpp.
|
|
|
|
logger = new AndroidLogger();
|
|
|
|
logman->AddListener(logger);
|
2016-01-10 12:08:19 +00:00
|
|
|
#endif
|
2016-09-10 15:16:53 +00:00
|
|
|
|
2018-03-24 11:51:54 +00:00
|
|
|
if (System_GetPropertyBool(SYSPROP_SUPPORTS_PERMISSIONS)) {
|
|
|
|
if (System_GetPermissionStatus(SYSTEM_PERMISSION_STORAGE) != PERMISSION_STATUS_GRANTED) {
|
|
|
|
System_AskForPermission(SYSTEM_PERMISSION_STORAGE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-01 20:45:35 +00:00
|
|
|
I18NCategory *des = GetI18NCategory("DesktopUI");
|
2013-11-29 15:33:17 +00:00
|
|
|
// Note to translators: do not translate this/add this to PPSSPP-lang's files.
|
|
|
|
// It's intended to be custom for every user.
|
2013-08-30 17:52:15 +00:00
|
|
|
// Only add it to your own personal copies of PPSSPP.
|
2018-06-09 18:34:12 +00:00
|
|
|
#if defined(USING_WIN_UI) && !PPSSPP_PLATFORM(UWP)
|
2013-08-31 08:36:52 +00:00
|
|
|
// TODO: Could allow a setting to specify a font file to load?
|
|
|
|
// TODO: Make this a constant if we can sanely load the font on other systems?
|
|
|
|
AddFontResourceEx(L"assets/Roboto-Condensed.ttf", FR_PRIVATE, NULL);
|
2015-10-04 22:11:36 +00:00
|
|
|
// The font goes by two names, let's allow either one.
|
|
|
|
if (CheckFontIsUsable(L"Roboto Condensed")) {
|
|
|
|
g_Config.sFont = des->T("Font", "Roboto Condensed");
|
|
|
|
} else {
|
|
|
|
g_Config.sFont = des->T("Font", "Roboto");
|
|
|
|
}
|
2018-06-09 18:34:12 +00:00
|
|
|
#elif defined(USING_QT_UI)
|
|
|
|
size_t fontSize = 0;
|
|
|
|
uint8_t *fontData = VFSReadFile("Roboto-Condensed.ttf", &fontSize);
|
|
|
|
if (fontData) {
|
|
|
|
int fontID = QFontDatabase::addApplicationFontFromData(QByteArray((const char *)fontData, fontSize));
|
|
|
|
delete [] fontData;
|
|
|
|
|
|
|
|
QStringList fontsFound = QFontDatabase::applicationFontFamilies(fontID);
|
|
|
|
if (fontsFound.size() >= 1) {
|
|
|
|
// Might be "Roboto" or "Roboto Condensed".
|
|
|
|
g_Config.sFont = des->T("Font", fontsFound.at(0).toUtf8().constData());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Let's try for it being a system font.
|
|
|
|
g_Config.sFont = des->T("Font", "Roboto Condensed");
|
|
|
|
}
|
2013-08-30 18:19:03 +00:00
|
|
|
#endif
|
2013-04-18 09:58:54 +00:00
|
|
|
|
2016-05-28 04:25:05 +00:00
|
|
|
if (!boot_filename.empty() && stateToLoad != NULL) {
|
2018-06-15 00:52:44 +00:00
|
|
|
SaveState::Load(stateToLoad, [](SaveState::Status status, const std::string &message, void *) {
|
2016-05-28 04:25:05 +00:00
|
|
|
if (!message.empty()) {
|
2018-06-15 00:52:44 +00:00
|
|
|
osm.Show(message, status == SaveState::Status::SUCCESS ? 2.0 : 5.0);
|
2016-05-28 04:25:05 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2013-10-17 14:44:09 +00:00
|
|
|
|
2012-11-04 10:54:45 +00:00
|
|
|
screenManager = new ScreenManager();
|
2013-07-27 11:07:34 +00:00
|
|
|
if (skipLogo) {
|
|
|
|
screenManager->switchScreen(new EmuScreen(boot_filename));
|
|
|
|
} else {
|
2013-11-03 03:18:12 +00:00
|
|
|
screenManager->switchScreen(new LogoScreen());
|
2013-07-27 11:07:34 +00:00
|
|
|
}
|
2013-10-31 10:06:54 +00:00
|
|
|
|
2018-04-21 20:54:44 +00:00
|
|
|
if (g_Config.bRemoteShareOnStartup && g_Config.bRemoteDebuggerOnStartup)
|
2018-04-21 20:51:18 +00:00
|
|
|
StartWebServer(WebServerFlags::ALL);
|
2018-04-21 20:54:44 +00:00
|
|
|
else if (g_Config.bRemoteShareOnStartup)
|
|
|
|
StartWebServer(WebServerFlags::DISCS);
|
|
|
|
else if (g_Config.bRemoteDebuggerOnStartup)
|
|
|
|
StartWebServer(WebServerFlags::DEBUGGER);
|
2017-12-11 02:30:28 +00:00
|
|
|
|
2013-10-31 10:06:54 +00:00
|
|
|
std::string sysName = System_GetProperty(SYSPROP_NAME);
|
|
|
|
isOuya = KeyMap::IsOuya(sysName);
|
2014-03-01 15:09:16 +00:00
|
|
|
|
|
|
|
#if !defined(MOBILE_DEVICE) && defined(USING_QT_UI)
|
2018-06-09 23:28:15 +00:00
|
|
|
MainWindow *mainWindow = new MainWindow(nullptr, g_Config.bFullScreen);
|
2014-03-01 15:09:16 +00:00
|
|
|
mainWindow->show();
|
2017-04-29 19:02:07 +00:00
|
|
|
if (host == nullptr) {
|
|
|
|
host = new QtHost(mainWindow);
|
|
|
|
}
|
2014-03-01 15:09:16 +00:00
|
|
|
#endif
|
2014-06-29 08:09:15 +00:00
|
|
|
|
|
|
|
// We do this here, instead of in NativeInitGraphics, because the display may be reset.
|
|
|
|
// When it's reset we don't want to forget all our managed things.
|
2018-09-01 20:57:20 +00:00
|
|
|
CheckFailedGPUBackends();
|
2016-01-06 06:37:28 +00:00
|
|
|
SetGPUBackend((GPUBackend) g_Config.iGPUBackend);
|
2018-09-07 01:31:09 +00:00
|
|
|
renderCounter = 0;
|
2017-04-15 23:30:37 +00:00
|
|
|
|
|
|
|
// Must be done restarting by now.
|
|
|
|
restarting = false;
|
2013-07-16 20:50:53 +00:00
|
|
|
}
|
|
|
|
|
2017-03-26 16:51:33 +00:00
|
|
|
static UI::Style MakeStyle(uint32_t fg, uint32_t bg) {
|
|
|
|
UI::Style s;
|
|
|
|
s.background = UI::Drawable(bg);
|
|
|
|
s.fgColor = fg;
|
2014-08-17 10:19:04 +00:00
|
|
|
|
2017-03-26 16:51:33 +00:00
|
|
|
return s;
|
|
|
|
}
|
2013-07-16 20:50:53 +00:00
|
|
|
|
2017-03-26 16:51:33 +00:00
|
|
|
static void UIThemeInit() {
|
2018-06-09 18:34:12 +00:00
|
|
|
#if (defined(USING_WIN_UI) && !PPSSPP_PLATFORM(UWP)) || defined(USING_QT_UI)
|
2013-08-31 08:36:52 +00:00
|
|
|
ui_theme.uiFont = UI::FontStyle(UBUNTU24, g_Config.sFont.c_str(), 22);
|
|
|
|
ui_theme.uiFontSmall = UI::FontStyle(UBUNTU24, g_Config.sFont.c_str(), 15);
|
|
|
|
ui_theme.uiFontSmaller = UI::FontStyle(UBUNTU24, g_Config.sFont.c_str(), 12);
|
2013-08-30 18:19:03 +00:00
|
|
|
#else
|
|
|
|
ui_theme.uiFont = UI::FontStyle(UBUNTU24, "", 20);
|
|
|
|
ui_theme.uiFontSmall = UI::FontStyle(UBUNTU24, "", 14);
|
|
|
|
ui_theme.uiFontSmaller = UI::FontStyle(UBUNTU24, "", 11);
|
|
|
|
#endif
|
2013-08-30 12:47:28 +00:00
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
ui_theme.checkOn = I_CHECKEDBOX;
|
|
|
|
ui_theme.checkOff = I_SQUARE;
|
2013-08-19 22:49:25 +00:00
|
|
|
ui_theme.whiteImage = I_SOLIDWHITE;
|
2013-07-18 08:26:29 +00:00
|
|
|
ui_theme.sliderKnob = I_CIRCLE;
|
2013-08-16 14:48:43 +00:00
|
|
|
ui_theme.dropShadow4Grid = I_DROP_SHADOW;
|
|
|
|
|
2017-03-26 16:51:33 +00:00
|
|
|
ui_theme.itemStyle = MakeStyle(g_Config.uItemStyleFg, g_Config.uItemStyleBg);
|
|
|
|
ui_theme.itemFocusedStyle = MakeStyle(g_Config.uItemFocusedStyleFg, g_Config.uItemFocusedStyleBg);
|
|
|
|
ui_theme.itemDownStyle = MakeStyle(g_Config.uItemDownStyleFg, g_Config.uItemDownStyleBg);
|
|
|
|
ui_theme.itemDisabledStyle = MakeStyle(g_Config.uItemDisabledStyleFg, g_Config.uItemDisabledStyleBg);
|
|
|
|
ui_theme.itemHighlightedStyle = MakeStyle(g_Config.uItemHighlightedStyleFg, g_Config.uItemHighlightedStyleBg);
|
|
|
|
|
|
|
|
ui_theme.buttonStyle = MakeStyle(g_Config.uButtonStyleFg, g_Config.uButtonStyleBg);
|
|
|
|
ui_theme.buttonFocusedStyle = MakeStyle(g_Config.uButtonFocusedStyleFg, g_Config.uButtonFocusedStyleBg);
|
|
|
|
ui_theme.buttonDownStyle = MakeStyle(g_Config.uButtonDownStyleFg, g_Config.uButtonDownStyleBg);
|
|
|
|
ui_theme.buttonDisabledStyle = MakeStyle(g_Config.uButtonDisabledStyleFg, g_Config.uButtonDisabledStyleBg);
|
|
|
|
ui_theme.buttonHighlightedStyle = MakeStyle(g_Config.uButtonHighlightedStyleFg, g_Config.uButtonHighlightedStyleBg);
|
|
|
|
|
|
|
|
ui_theme.headerStyle.fgColor = g_Config.uHeaderStyleFg;
|
|
|
|
ui_theme.infoStyle = MakeStyle(g_Config.uInfoStyleFg, g_Config.uInfoStyleBg);
|
|
|
|
|
|
|
|
ui_theme.popupTitle.fgColor = g_Config.uPopupTitleStyleFg;
|
|
|
|
ui_theme.popupStyle = MakeStyle(g_Config.uPopupStyleFg, g_Config.uPopupStyleBg);
|
|
|
|
}
|
|
|
|
|
2017-05-16 12:09:15 +00:00
|
|
|
void RenderOverlays(UIContext *dc, void *userdata);
|
|
|
|
|
2017-11-29 17:53:52 +00:00
|
|
|
bool NativeInitGraphics(GraphicsContext *graphicsContext) {
|
2017-03-26 16:51:33 +00:00
|
|
|
ILOG("NativeInitGraphics");
|
2017-11-29 17:53:52 +00:00
|
|
|
_assert_msg_(G3D, graphicsContext, "No graphics context!");
|
2017-03-26 16:51:33 +00:00
|
|
|
|
|
|
|
using namespace Draw;
|
|
|
|
Core_SetGraphicsContext(graphicsContext);
|
|
|
|
g_draw = graphicsContext->GetDrawContext();
|
2017-11-29 17:53:52 +00:00
|
|
|
_assert_msg_(G3D, g_draw, "No draw context available!");
|
2017-03-26 16:51:33 +00:00
|
|
|
|
|
|
|
ui_draw2d.SetAtlas(&ui_atlas);
|
|
|
|
ui_draw2d_front.SetAtlas(&ui_atlas);
|
|
|
|
|
|
|
|
UIThemeInit();
|
2013-12-10 22:19:37 +00:00
|
|
|
|
2013-03-30 18:23:20 +00:00
|
|
|
uiContext = new UIContext();
|
2013-06-01 21:34:50 +00:00
|
|
|
uiContext->theme = &ui_theme;
|
2014-08-17 10:19:04 +00:00
|
|
|
|
2017-01-30 13:33:38 +00:00
|
|
|
Draw::InputLayout *inputLayout = ui_draw2d.CreateInputLayout(g_draw);
|
|
|
|
Draw::BlendState *blendNormal = g_draw->CreateBlendState({ true, 0xF, BlendFactor::SRC_ALPHA, BlendFactor::ONE_MINUS_SRC_ALPHA });
|
|
|
|
Draw::DepthStencilState *depth = g_draw->CreateDepthStencilState({ false, false, Comparison::LESS });
|
|
|
|
Draw::RasterState *rasterNoCull = g_draw->CreateRasterState({});
|
2016-12-26 16:03:01 +00:00
|
|
|
|
|
|
|
PipelineDesc colorDesc{
|
|
|
|
Primitive::TRIANGLE_LIST,
|
2017-01-30 13:33:38 +00:00
|
|
|
{ g_draw->GetVshaderPreset(VS_COLOR_2D), g_draw->GetFshaderPreset(FS_COLOR_2D) },
|
2017-02-08 11:55:58 +00:00
|
|
|
inputLayout, depth, blendNormal, rasterNoCull, &vsColBufDesc,
|
2016-12-26 16:03:01 +00:00
|
|
|
};
|
|
|
|
PipelineDesc texColorDesc{
|
|
|
|
Primitive::TRIANGLE_LIST,
|
2017-01-30 13:33:38 +00:00
|
|
|
{ g_draw->GetVshaderPreset(VS_TEXTURE_COLOR_2D), g_draw->GetFshaderPreset(FS_TEXTURE_COLOR_2D) },
|
2017-02-08 11:55:58 +00:00
|
|
|
inputLayout, depth, blendNormal, rasterNoCull, &vsTexColBufDesc,
|
2016-12-26 16:03:01 +00:00
|
|
|
};
|
2016-12-26 12:42:53 +00:00
|
|
|
|
2017-01-30 13:33:38 +00:00
|
|
|
colorPipeline = g_draw->CreateGraphicsPipeline(colorDesc);
|
|
|
|
texColorPipeline = g_draw->CreateGraphicsPipeline(texColorDesc);
|
2016-12-26 12:42:53 +00:00
|
|
|
|
2018-06-01 19:16:07 +00:00
|
|
|
_assert_(colorPipeline);
|
|
|
|
_assert_(texColorPipeline);
|
|
|
|
|
2017-02-17 18:22:41 +00:00
|
|
|
// Release these now, reference counting should ensure that they get completely released
|
|
|
|
// once we delete both pipelines.
|
2016-12-26 16:03:01 +00:00
|
|
|
inputLayout->Release();
|
|
|
|
rasterNoCull->Release();
|
|
|
|
blendNormal->Release();
|
|
|
|
depth->Release();
|
|
|
|
|
2017-01-30 13:33:38 +00:00
|
|
|
ui_draw2d.Init(g_draw, texColorPipeline);
|
|
|
|
ui_draw2d_front.Init(g_draw, texColorPipeline);
|
2016-12-26 16:03:01 +00:00
|
|
|
|
2017-01-30 13:33:38 +00:00
|
|
|
uiContext->Init(g_draw, texColorPipeline, colorPipeline, &ui_draw2d, &ui_draw2d_front);
|
2016-12-26 16:03:01 +00:00
|
|
|
RasterStateDesc desc;
|
|
|
|
desc.cull = CullMode::NONE;
|
2016-12-27 14:52:03 +00:00
|
|
|
desc.frontFace = Facing::CCW;
|
2016-12-26 16:03:01 +00:00
|
|
|
|
2013-08-30 16:15:45 +00:00
|
|
|
if (uiContext->Text())
|
|
|
|
uiContext->Text()->SetFont("Tahoma", 20, 0);
|
2014-02-10 14:14:45 +00:00
|
|
|
|
2013-03-30 18:23:20 +00:00
|
|
|
screenManager->setUIContext(uiContext);
|
2017-01-30 13:33:38 +00:00
|
|
|
screenManager->setDrawContext(g_draw);
|
2017-05-16 12:09:15 +00:00
|
|
|
screenManager->setPostRenderCallback(&RenderOverlays, nullptr);
|
2018-03-27 21:10:33 +00:00
|
|
|
screenManager->deviceRestored();
|
2013-03-30 18:23:20 +00:00
|
|
|
|
2014-06-22 12:23:06 +00:00
|
|
|
#ifdef _WIN32
|
2015-01-24 12:50:27 +00:00
|
|
|
winAudioBackend = CreateAudioBackend((AudioBackendType)g_Config.iAudioBackend);
|
2017-02-24 23:25:46 +00:00
|
|
|
#if PPSSPP_PLATFORM(UWP)
|
2017-03-01 11:51:06 +00:00
|
|
|
winAudioBackend->Init(0, &Win32Mix, 44100);
|
2017-02-24 23:25:46 +00:00
|
|
|
#else
|
2015-01-11 20:00:56 +00:00
|
|
|
winAudioBackend->Init(MainWindow::GetHWND(), &Win32Mix, 44100);
|
2017-02-24 23:25:46 +00:00
|
|
|
#endif
|
2014-06-22 12:23:06 +00:00
|
|
|
#endif
|
2016-02-14 21:07:10 +00:00
|
|
|
|
|
|
|
g_gameInfoCache = new GameInfoCache();
|
2017-11-10 12:13:56 +00:00
|
|
|
|
2017-12-07 13:56:19 +00:00
|
|
|
if (gpu)
|
|
|
|
gpu->DeviceRestore();
|
|
|
|
|
2017-11-10 12:13:56 +00:00
|
|
|
g_graphicsInited = true;
|
2017-03-18 14:20:36 +00:00
|
|
|
ILOG("NativeInitGraphics completed");
|
2017-11-29 17:53:52 +00:00
|
|
|
return true;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2013-07-16 20:50:53 +00:00
|
|
|
void NativeShutdownGraphics() {
|
2018-03-27 21:10:33 +00:00
|
|
|
screenManager->deviceLost();
|
|
|
|
|
2017-12-07 13:56:19 +00:00
|
|
|
if (gpu)
|
|
|
|
gpu->DeviceLost();
|
|
|
|
|
2017-11-10 12:13:56 +00:00
|
|
|
g_graphicsInited = false;
|
2017-03-18 14:20:36 +00:00
|
|
|
ILOG("NativeShutdownGraphics");
|
|
|
|
|
2014-06-22 12:23:06 +00:00
|
|
|
#ifdef _WIN32
|
2015-01-11 20:00:56 +00:00
|
|
|
delete winAudioBackend;
|
2017-03-25 22:01:33 +00:00
|
|
|
winAudioBackend = nullptr;
|
2014-06-22 12:23:06 +00:00
|
|
|
#endif
|
2014-06-22 12:17:57 +00:00
|
|
|
|
2016-02-14 21:07:10 +00:00
|
|
|
delete g_gameInfoCache;
|
|
|
|
g_gameInfoCache = nullptr;
|
2013-07-16 20:50:53 +00:00
|
|
|
|
2017-03-25 22:01:33 +00:00
|
|
|
UIBackgroundShutdown();
|
|
|
|
|
2013-07-16 20:50:53 +00:00
|
|
|
delete uiContext;
|
2017-03-25 22:01:33 +00:00
|
|
|
uiContext = nullptr;
|
2013-07-16 20:50:53 +00:00
|
|
|
|
|
|
|
ui_draw2d.Shutdown();
|
|
|
|
ui_draw2d_front.Shutdown();
|
|
|
|
|
2018-01-20 20:47:16 +00:00
|
|
|
if (colorPipeline) {
|
|
|
|
colorPipeline->Release();
|
|
|
|
colorPipeline = nullptr;
|
|
|
|
}
|
|
|
|
if (texColorPipeline) {
|
|
|
|
texColorPipeline->Release();
|
|
|
|
texColorPipeline = nullptr;
|
|
|
|
}
|
2017-03-18 14:20:36 +00:00
|
|
|
|
|
|
|
ILOG("NativeShutdownGraphics done");
|
2013-07-16 20:50:53 +00:00
|
|
|
}
|
|
|
|
|
2013-06-22 20:27:59 +00:00
|
|
|
void TakeScreenshot() {
|
|
|
|
g_TakeScreenshot = false;
|
2014-08-17 14:07:14 +00:00
|
|
|
|
2014-12-21 07:14:46 +00:00
|
|
|
#if defined(_WIN32) || (defined(USING_QT_UI) && !defined(MOBILE_DEVICE))
|
2014-12-29 04:52:44 +00:00
|
|
|
std::string path = GetSysDirectory(DIRECTORY_SCREENSHOT);
|
|
|
|
while (path.length() > 0 && path.back() == '/') {
|
|
|
|
path.resize(path.size() - 1);
|
|
|
|
}
|
|
|
|
if (!File::Exists(path)) {
|
|
|
|
File::CreateDir(path);
|
|
|
|
}
|
2013-06-22 20:27:59 +00:00
|
|
|
|
|
|
|
// First, find a free filename.
|
|
|
|
int i = 0;
|
|
|
|
|
2017-06-03 03:54:28 +00:00
|
|
|
std::string gameId = g_paramSFO.GetDiscID();
|
2014-02-17 22:43:58 +00:00
|
|
|
|
2014-09-13 22:14:11 +00:00
|
|
|
char filename[2048];
|
2013-07-07 18:56:18 +00:00
|
|
|
while (i < 10000){
|
2014-02-17 22:43:58 +00:00
|
|
|
if (g_Config.bScreenshotsAsPNG)
|
2014-12-29 04:52:44 +00:00
|
|
|
snprintf(filename, sizeof(filename), "%s/%s_%05d.png", path.c_str(), gameId.c_str(), i);
|
2013-07-07 18:56:18 +00:00
|
|
|
else
|
2014-12-29 04:52:44 +00:00
|
|
|
snprintf(filename, sizeof(filename), "%s/%s_%05d.jpg", path.c_str(), gameId.c_str(), i);
|
2013-06-22 20:27:59 +00:00
|
|
|
FileInfo info;
|
2014-02-17 22:43:58 +00:00
|
|
|
if (!getFileInfo(filename, &info))
|
2013-06-22 20:27:59 +00:00
|
|
|
break;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
2017-12-21 23:44:49 +00:00
|
|
|
bool success = TakeGameScreenshot(filename, g_Config.bScreenshotsAsPNG ? ScreenshotFormat::PNG : ScreenshotFormat::JPG, SCREENSHOT_OUTPUT);
|
2014-11-02 21:29:44 +00:00
|
|
|
if (success) {
|
|
|
|
osm.Show(filename);
|
|
|
|
} else {
|
|
|
|
I18NCategory *err = GetI18NCategory("Error");
|
|
|
|
osm.Show(err->T("Could not save screenshot file"));
|
|
|
|
}
|
2013-06-22 20:27:59 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-16 12:09:15 +00:00
|
|
|
void RenderOverlays(UIContext *dc, void *userdata) {
|
2013-11-29 15:33:17 +00:00
|
|
|
// Thin bar at the top of the screen like Chrome.
|
|
|
|
std::vector<float> progress = g_DownloadManager.GetCurrentProgress();
|
2017-06-02 03:40:21 +00:00
|
|
|
if (!progress.empty()) {
|
|
|
|
static const uint32_t colors[4] = {
|
|
|
|
0xFFFFFFFF,
|
|
|
|
0xFFCCCCCC,
|
|
|
|
0xFFAAAAAA,
|
|
|
|
0xFF777777,
|
|
|
|
};
|
|
|
|
|
|
|
|
dc->Begin();
|
|
|
|
int h = 5;
|
|
|
|
for (size_t i = 0; i < progress.size(); i++) {
|
|
|
|
float barWidth = 10 + (dc->GetBounds().w - 10) * progress[i];
|
|
|
|
Bounds bounds(0, h * i, barWidth, h);
|
|
|
|
UI::Drawable solid(colors[i & 3]);
|
|
|
|
dc->FillRect(solid, bounds);
|
|
|
|
}
|
|
|
|
dc->End();
|
|
|
|
dc->Flush();
|
2017-05-16 12:09:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (g_TakeScreenshot) {
|
|
|
|
TakeScreenshot();
|
2013-11-29 15:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-31 15:59:40 +00:00
|
|
|
void NativeRender(GraphicsContext *graphicsContext) {
|
2013-11-20 13:42:48 +00:00
|
|
|
g_GameManager.Update();
|
2017-05-16 12:09:15 +00:00
|
|
|
|
2014-02-10 14:14:45 +00:00
|
|
|
float xres = dp_xres;
|
|
|
|
float yres = dp_yres;
|
|
|
|
|
2014-02-10 11:38:23 +00:00
|
|
|
// Apply the UIContext bounds as a 2D transformation matrix.
|
2017-05-16 12:09:15 +00:00
|
|
|
// TODO: This should be moved into the draw context...
|
2012-11-04 09:56:22 +00:00
|
|
|
Matrix4x4 ortho;
|
2016-01-03 11:37:05 +00:00
|
|
|
switch (GetGPUBackend()) {
|
|
|
|
case GPUBackend::VULKAN:
|
|
|
|
ortho.setOrthoD3D(0.0f, xres, 0, yres, -1.0f, 1.0f);
|
|
|
|
break;
|
|
|
|
case GPUBackend::DIRECT3D9:
|
2014-08-23 08:49:40 +00:00
|
|
|
ortho.setOrthoD3D(0.0f, xres, yres, 0.0f, -1.0f, 1.0f);
|
2014-08-25 19:33:24 +00:00
|
|
|
Matrix4x4 translation;
|
2017-12-03 14:58:16 +00:00
|
|
|
// Account for the small window adjustment.
|
|
|
|
translation.setTranslation(Vec3(-0.5f * g_dpi_scale_x / g_dpi_scale_real_x, -0.5f * g_dpi_scale_y / g_dpi_scale_real_y, 0.0f));
|
2014-08-25 19:33:24 +00:00
|
|
|
ortho = translation * ortho;
|
2016-01-03 11:37:05 +00:00
|
|
|
break;
|
2017-02-08 16:09:59 +00:00
|
|
|
case GPUBackend::DIRECT3D11:
|
|
|
|
ortho.setOrthoD3D(0.0f, xres, yres, 0.0f, -1.0f, 1.0f);
|
|
|
|
break;
|
2016-01-03 11:37:05 +00:00
|
|
|
case GPUBackend::OPENGL:
|
2014-08-23 08:49:40 +00:00
|
|
|
ortho.setOrtho(0.0f, xres, yres, 0.0f, -1.0f, 1.0f);
|
2016-01-03 11:37:05 +00:00
|
|
|
break;
|
2014-08-23 08:49:40 +00:00
|
|
|
}
|
2014-02-10 14:55:21 +00:00
|
|
|
|
2017-03-05 00:39:26 +00:00
|
|
|
if (g_display_rotation != DisplayRotation::ROTATE_0) {
|
|
|
|
ortho = ortho * g_display_rot_matrix;
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:28:24 +00:00
|
|
|
ui_draw2d.PushDrawMatrix(ortho);
|
|
|
|
ui_draw2d_front.PushDrawMatrix(ortho);
|
2014-08-17 10:19:04 +00:00
|
|
|
|
2017-05-16 12:09:15 +00:00
|
|
|
// All actual rendering happen in here.
|
2012-11-04 10:54:45 +00:00
|
|
|
screenManager->render();
|
2013-11-15 12:11:44 +00:00
|
|
|
if (screenManager->getUIContext()->Text()) {
|
2013-08-30 16:15:45 +00:00
|
|
|
screenManager->getUIContext()->Text()->OncePerFrame();
|
2013-11-15 12:11:44 +00:00
|
|
|
}
|
2013-06-22 20:27:59 +00:00
|
|
|
|
2015-02-01 17:04:06 +00:00
|
|
|
if (resized) {
|
|
|
|
resized = false;
|
2017-02-23 08:25:33 +00:00
|
|
|
|
|
|
|
if (uiContext) {
|
|
|
|
// Modifying the bounds here can be used to "inset" the whole image to gain borders for TV overscan etc.
|
|
|
|
// The UI now supports any offset but not the EmuScreen yet.
|
|
|
|
uiContext->SetBounds(Bounds(0, 0, dp_xres, dp_yres));
|
|
|
|
// uiContext->SetBounds(Bounds(dp_xres/2, 0, dp_xres / 2, dp_yres / 2));
|
|
|
|
|
|
|
|
|
|
|
|
// OSX 10.6 and SDL 1.2 bug.
|
|
|
|
#if defined(__APPLE__) && !defined(USING_QT_UI)
|
|
|
|
static int dp_xres_old = dp_xres;
|
|
|
|
if (dp_xres != dp_xres_old) {
|
|
|
|
// uiTexture->Load("ui_atlas.zim");
|
|
|
|
dp_xres_old = dp_xres;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-12-07 13:56:19 +00:00
|
|
|
// Test lost/restore on PC
|
2017-11-10 14:10:36 +00:00
|
|
|
#if 0
|
2017-12-07 13:56:19 +00:00
|
|
|
if (gpu) {
|
|
|
|
gpu->DeviceLost();
|
|
|
|
gpu->DeviceRestore();
|
|
|
|
}
|
2017-11-10 14:10:36 +00:00
|
|
|
#endif
|
|
|
|
|
2017-12-07 13:56:19 +00:00
|
|
|
graphicsContext->Resize();
|
|
|
|
screenManager->resized();
|
|
|
|
|
2015-12-31 15:59:40 +00:00
|
|
|
// TODO: Move this to new GraphicsContext objects for each backend.
|
2015-10-25 10:19:41 +00:00
|
|
|
#ifndef _WIN32
|
2016-01-06 06:37:28 +00:00
|
|
|
if (GetGPUBackend() == GPUBackend::OPENGL) {
|
2015-10-25 10:19:41 +00:00
|
|
|
PSP_CoreParameter().pixelWidth = pixel_xres;
|
|
|
|
PSP_CoreParameter().pixelHeight = pixel_yres;
|
2017-04-28 03:03:50 +00:00
|
|
|
NativeMessageReceived("gpu_resized", "");
|
2015-02-01 17:04:06 +00:00
|
|
|
}
|
2015-12-31 15:59:40 +00:00
|
|
|
#endif
|
2015-02-01 17:04:06 +00:00
|
|
|
}
|
2017-03-19 21:28:24 +00:00
|
|
|
|
|
|
|
ui_draw2d.PopDrawMatrix();
|
|
|
|
ui_draw2d_front.PopDrawMatrix();
|
2018-09-01 20:57:20 +00:00
|
|
|
|
|
|
|
if (renderCounter < 10 && ++renderCounter == 10) {
|
|
|
|
// We're rendering fine, clear out failure info.
|
|
|
|
ClearFailedGPUBackends();
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2014-05-19 21:29:35 +00:00
|
|
|
void HandleGlobalMessage(const std::string &msg, const std::string &value) {
|
|
|
|
if (msg == "inputDeviceConnected") {
|
|
|
|
KeyMap::NotifyPadConnected(value);
|
|
|
|
}
|
2016-02-21 03:23:00 +00:00
|
|
|
if (msg == "inputbox_completed") {
|
|
|
|
SplitString(value, ':', inputboxValue);
|
2016-10-26 17:06:41 +00:00
|
|
|
std::string setString = inputboxValue.size() > 1 ? inputboxValue[1] : "";
|
2016-02-21 05:12:35 +00:00
|
|
|
if (inputboxValue[0] == "IP")
|
2016-10-26 17:06:41 +00:00
|
|
|
g_Config.proAdhocServer = setString;
|
2017-11-16 12:33:10 +00:00
|
|
|
else if (inputboxValue[0] == "nickname")
|
2016-10-26 17:06:41 +00:00
|
|
|
g_Config.sNickName = setString;
|
2017-11-16 12:33:10 +00:00
|
|
|
else if (inputboxValue[0] == "remoteiso_subdir")
|
|
|
|
g_Config.sRemoteISOSubdir = setString;
|
|
|
|
else if (inputboxValue[0] == "remoteiso_server")
|
|
|
|
g_Config.sLastRemoteISOServer = setString;
|
2016-02-21 03:23:00 +00:00
|
|
|
inputboxValue.clear();
|
|
|
|
}
|
2017-03-25 22:08:43 +00:00
|
|
|
if (msg == "bgImage_updated") {
|
2017-04-29 18:06:37 +00:00
|
|
|
if (!value.empty()) {
|
|
|
|
std::string dest = GetSysDirectory(DIRECTORY_SYSTEM) + (endsWithNoCase(value, ".jpg") ? "background.jpg" : "background.png");
|
|
|
|
File::Copy(value, dest);
|
|
|
|
}
|
2017-03-25 22:08:43 +00:00
|
|
|
UIBackgroundShutdown();
|
|
|
|
UIBackgroundInit(*uiContext);
|
|
|
|
}
|
2016-05-28 03:53:20 +00:00
|
|
|
if (msg == "savestate_displayslot") {
|
|
|
|
I18NCategory *sy = GetI18NCategory("System");
|
|
|
|
std::string msg = StringFromFormat("%s: %d", sy->T("Savestate Slot"), SaveState::GetCurrentSlot() + 1);
|
2016-05-28 03:53:58 +00:00
|
|
|
// Show for the same duration as the preview.
|
2016-05-28 03:55:44 +00:00
|
|
|
osm.Show(msg, 2.0f, 0xFFFFFF, -1, true, "savestate_slot");
|
2016-05-28 03:53:20 +00:00
|
|
|
}
|
2017-04-28 03:03:50 +00:00
|
|
|
if (msg == "gpu_resized" || msg == "gpu_clearCache") {
|
2016-09-29 05:35:09 +00:00
|
|
|
if (gpu) {
|
|
|
|
gpu->ClearCacheNextFrame();
|
|
|
|
gpu->Resized();
|
|
|
|
}
|
|
|
|
Reporting::UpdateConfig();
|
|
|
|
}
|
2016-07-25 00:04:06 +00:00
|
|
|
if (msg == "core_powerSaving") {
|
|
|
|
if (value != "false") {
|
|
|
|
I18NCategory *sy = GetI18NCategory("System");
|
2016-10-12 09:13:16 +00:00
|
|
|
#ifdef __ANDROID__
|
2016-10-11 00:32:25 +00:00
|
|
|
osm.Show(sy->T("WARNING: Android battery save mode is on"), 2.0f, 0xFFFFFF, -1, true, "core_powerSaving");
|
|
|
|
#else
|
2016-07-25 00:04:06 +00:00
|
|
|
osm.Show(sy->T("WARNING: Battery save mode is on"), 2.0f, 0xFFFFFF, -1, true, "core_powerSaving");
|
2016-10-11 00:32:25 +00:00
|
|
|
#endif
|
2016-07-25 00:04:06 +00:00
|
|
|
}
|
|
|
|
Core_SetPowerSaving(value != "false");
|
|
|
|
}
|
2018-03-24 11:51:54 +00:00
|
|
|
if (msg == "permission_granted" && value == "storage") {
|
2018-05-10 17:32:28 +00:00
|
|
|
#ifdef __ANDROID__
|
|
|
|
CreateDirectoriesAndroid();
|
|
|
|
#endif
|
2018-03-24 11:51:54 +00:00
|
|
|
// We must have failed to load the config before, so load it now to avoid overwriting the old config
|
|
|
|
// with a freshly generated one.
|
|
|
|
ILOG("Reloading config after storage permission grant.");
|
|
|
|
g_Config.Load();
|
2018-04-29 18:45:32 +00:00
|
|
|
PostLoadConfig();
|
2018-03-24 11:51:54 +00:00
|
|
|
}
|
2014-05-19 21:29:35 +00:00
|
|
|
}
|
|
|
|
|
2017-03-15 05:01:18 +00:00
|
|
|
void NativeUpdate() {
|
2015-05-15 15:14:54 +00:00
|
|
|
PROFILE_END_FRAME();
|
|
|
|
|
2017-12-10 10:28:27 +00:00
|
|
|
std::vector<PendingMessage> toProcess;
|
2013-03-29 19:51:14 +00:00
|
|
|
{
|
2017-02-27 20:57:46 +00:00
|
|
|
std::lock_guard<std::mutex> lock(pendingMutex);
|
2017-12-10 10:28:27 +00:00
|
|
|
toProcess = std::move(pendingMessages);
|
2014-02-13 16:47:02 +00:00
|
|
|
pendingMessages.clear();
|
2013-03-29 19:51:14 +00:00
|
|
|
}
|
|
|
|
|
2017-12-10 10:28:27 +00:00
|
|
|
for (size_t i = 0; i < toProcess.size(); i++) {
|
|
|
|
HandleGlobalMessage(toProcess[i].msg, toProcess[i].value);
|
|
|
|
screenManager->sendMessage(toProcess[i].msg.c_str(), toProcess[i].value.c_str());
|
|
|
|
}
|
|
|
|
|
2013-11-26 13:04:29 +00:00
|
|
|
g_DownloadManager.Update();
|
2017-03-15 05:01:18 +00:00
|
|
|
screenManager->update();
|
2018-08-12 22:08:56 +00:00
|
|
|
|
|
|
|
g_Discord.Update();
|
2013-07-11 12:29:56 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
bool NativeIsAtTopLevel() {
|
2018-05-26 15:50:37 +00:00
|
|
|
// This might need some synchronization?
|
|
|
|
if (!screenManager) {
|
|
|
|
ELOG("No screen manager active");
|
|
|
|
return false;
|
|
|
|
}
|
2013-09-04 08:51:14 +00:00
|
|
|
Screen *currentScreen = screenManager->topScreen();
|
|
|
|
if (currentScreen) {
|
|
|
|
bool top = currentScreen->isTopLevel();
|
|
|
|
ILOG("Screen toplevel: %i", (int)top);
|
|
|
|
return currentScreen->isTopLevel();
|
|
|
|
} else {
|
|
|
|
ELOG("No current screen");
|
|
|
|
return false;
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2014-06-15 10:13:59 +00:00
|
|
|
bool NativeTouch(const TouchInput &touch) {
|
|
|
|
if (screenManager) {
|
2017-05-03 22:30:04 +00:00
|
|
|
// Brute force prevent NaNs from getting into the UI system
|
|
|
|
if (my_isnan(touch.x) || my_isnan(touch.y)) {
|
|
|
|
return false;
|
|
|
|
}
|
2013-06-10 21:47:49 +00:00
|
|
|
screenManager->touch(touch);
|
2014-06-15 10:13:59 +00:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
2014-06-15 10:13:59 +00:00
|
|
|
bool NativeKey(const KeyInput &key) {
|
2013-10-31 10:06:54 +00:00
|
|
|
// ILOG("Key code: %i flags: %i", key.keyCode, key.flags);
|
2014-04-26 05:00:45 +00:00
|
|
|
#if !defined(MOBILE_DEVICE)
|
2014-04-26 05:54:32 +00:00
|
|
|
if (g_Config.bPauseExitsEmulator) {
|
|
|
|
static std::vector<int> pspKeys;
|
|
|
|
pspKeys.clear();
|
|
|
|
if (KeyMap::KeyToPspButton(key.deviceId, key.keyCode, &pspKeys)) {
|
|
|
|
if (std::find(pspKeys.begin(), pspKeys.end(), VIRTKEY_PAUSE) != pspKeys.end()) {
|
|
|
|
System_SendMessage("finish", "");
|
2014-06-15 10:13:59 +00:00
|
|
|
return true;
|
2014-04-26 05:54:32 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-26 05:00:45 +00:00
|
|
|
}
|
|
|
|
#endif
|
2014-06-15 11:04:59 +00:00
|
|
|
bool retval = false;
|
2013-07-06 17:08:59 +00:00
|
|
|
if (screenManager)
|
2014-06-15 11:04:59 +00:00
|
|
|
retval = screenManager->key(key);
|
|
|
|
return retval;
|
2013-07-06 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
2014-07-20 16:30:43 +00:00
|
|
|
bool NativeAxis(const AxisInput &axis) {
|
2013-11-11 11:52:04 +00:00
|
|
|
using namespace TiltEventProcessor;
|
|
|
|
|
2014-06-15 10:13:59 +00:00
|
|
|
// only handle tilt events if tilt is enabled.
|
2014-07-20 16:30:43 +00:00
|
|
|
if (g_Config.iTiltInputType == TILT_NULL) {
|
2014-06-15 10:13:59 +00:00
|
|
|
// if tilt events are disabled, then run it through the usual way.
|
2013-11-11 12:37:17 +00:00
|
|
|
if (screenManager) {
|
2014-07-20 16:30:43 +00:00
|
|
|
return screenManager->axis(axis);
|
2014-06-15 10:13:59 +00:00
|
|
|
} else {
|
|
|
|
return false;
|
2013-11-11 12:37:17 +00:00
|
|
|
}
|
2013-08-17 11:41:04 +00:00
|
|
|
}
|
2014-06-15 10:13:59 +00:00
|
|
|
|
2014-07-20 16:30:43 +00:00
|
|
|
// create the base coordinate tilt system from the calibration data.
|
|
|
|
// This is static for no particular reason, can be un-static'ed
|
2013-11-11 11:52:04 +00:00
|
|
|
static Tilt baseTilt;
|
2014-06-15 10:13:59 +00:00
|
|
|
baseTilt.x_ = g_Config.fTiltBaseX;
|
|
|
|
baseTilt.y_ = g_Config.fTiltBaseY;
|
2013-11-11 11:52:04 +00:00
|
|
|
|
2014-07-20 16:30:43 +00:00
|
|
|
// figure out what the current tilt orientation is by checking the axis event
|
|
|
|
// This is static, since we need to remember where we last were (in terms of orientation)
|
2013-11-11 11:52:04 +00:00
|
|
|
static Tilt currentTilt;
|
|
|
|
|
2014-07-20 16:30:43 +00:00
|
|
|
// x and y are flipped if we are in landscape orientation. The events are
|
|
|
|
// sent with respect to the portrait coordinate system, while we
|
|
|
|
// take all events in landscape.
|
|
|
|
// see [http://developer.android.com/guide/topics/sensors/sensors_overview.html] for details
|
|
|
|
bool portrait = dp_yres > dp_xres;
|
|
|
|
switch (axis.axisId) {
|
2013-11-11 11:52:04 +00:00
|
|
|
case JOYSTICK_AXIS_ACCELEROMETER_X:
|
2014-07-20 16:30:43 +00:00
|
|
|
if (portrait) {
|
|
|
|
currentTilt.x_ = axis.value;
|
|
|
|
} else {
|
|
|
|
currentTilt.y_ = axis.value;
|
|
|
|
}
|
2013-11-11 11:52:04 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case JOYSTICK_AXIS_ACCELEROMETER_Y:
|
2014-07-20 16:30:43 +00:00
|
|
|
if (portrait) {
|
|
|
|
currentTilt.y_ = axis.value;
|
|
|
|
} else {
|
|
|
|
currentTilt.x_ = axis.value;
|
|
|
|
}
|
2013-11-11 11:52:04 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case JOYSTICK_AXIS_ACCELEROMETER_Z:
|
|
|
|
//don't handle this now as only landscape is enabled.
|
|
|
|
//TODO: make this generic.
|
2014-06-15 10:13:59 +00:00
|
|
|
return false;
|
2013-11-11 11:52:04 +00:00
|
|
|
|
|
|
|
case JOYSTICK_AXIS_OUYA_UNKNOWN1:
|
|
|
|
case JOYSTICK_AXIS_OUYA_UNKNOWN2:
|
|
|
|
case JOYSTICK_AXIS_OUYA_UNKNOWN3:
|
|
|
|
case JOYSTICK_AXIS_OUYA_UNKNOWN4:
|
|
|
|
//Don't know how to handle these. Someone should figure it out.
|
|
|
|
//Does the Ouya even have an accelerometer / gyro? I can't find any reference to these
|
|
|
|
//in the Ouya docs...
|
2014-06-15 10:13:59 +00:00
|
|
|
return false;
|
2013-11-11 11:52:04 +00:00
|
|
|
|
|
|
|
default:
|
2015-04-03 09:48:58 +00:00
|
|
|
// Don't take over completely!
|
|
|
|
return screenManager->axis(axis);
|
2013-10-31 10:06:54 +00:00
|
|
|
}
|
2013-11-11 11:52:04 +00:00
|
|
|
|
|
|
|
//figure out the sensitivity of the tilt. (sensitivity is originally 0 - 100)
|
|
|
|
//We divide by 50, so that the rest of the 50 units can be used to overshoot the
|
|
|
|
//target. If you want control, you'd keep the sensitivity ~50.
|
|
|
|
//For games that don't need much control but need fast reactions,
|
|
|
|
//then a value of 70-80 is the way to go.
|
|
|
|
float xSensitivity = g_Config.iTiltSensitivityX / 50.0;
|
|
|
|
float ySensitivity = g_Config.iTiltSensitivityY / 50.0;
|
|
|
|
|
|
|
|
//now transform out current tilt to the calibrated coordinate system
|
|
|
|
Tilt trueTilt = GenTilt(baseTilt, currentTilt, g_Config.bInvertTiltX, g_Config.bInvertTiltY, g_Config.fDeadzoneRadius, xSensitivity, ySensitivity);
|
|
|
|
|
2015-12-21 07:03:34 +00:00
|
|
|
TranslateTiltToInput(trueTilt);
|
2014-06-15 10:13:59 +00:00
|
|
|
return true;
|
2013-07-06 17:08:59 +00:00
|
|
|
}
|
|
|
|
|
2013-06-01 21:34:50 +00:00
|
|
|
void NativeMessageReceived(const char *message, const char *value) {
|
2013-03-29 19:51:14 +00:00
|
|
|
// We can only have one message queued.
|
2017-02-27 20:57:46 +00:00
|
|
|
std::lock_guard<std::mutex> lock(pendingMutex);
|
2014-02-13 16:47:02 +00:00
|
|
|
PendingMessage pendingMessage;
|
|
|
|
pendingMessage.msg = message;
|
|
|
|
pendingMessage.value = value;
|
|
|
|
pendingMessages.push_back(pendingMessage);
|
2012-12-01 09:39:20 +00:00
|
|
|
}
|
|
|
|
|
2013-12-16 15:21:10 +00:00
|
|
|
void NativeResized() {
|
2017-02-23 08:25:33 +00:00
|
|
|
// NativeResized can come from any thread so we just set a flag, then process it later.
|
2017-11-10 12:13:56 +00:00
|
|
|
if (g_graphicsInited) {
|
|
|
|
resized = true;
|
|
|
|
} else {
|
|
|
|
ILOG("NativeResized ignored, not initialized");
|
|
|
|
}
|
2013-12-16 15:21:10 +00:00
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2017-04-15 23:30:37 +00:00
|
|
|
void NativeSetRestarting() {
|
|
|
|
restarting = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool NativeIsRestarting() {
|
|
|
|
return restarting;
|
|
|
|
}
|
|
|
|
|
2013-07-16 20:50:53 +00:00
|
|
|
void NativeShutdown() {
|
2018-06-01 19:16:07 +00:00
|
|
|
if (screenManager)
|
|
|
|
screenManager->shutdown();
|
2017-04-29 19:02:07 +00:00
|
|
|
delete screenManager;
|
|
|
|
screenManager = nullptr;
|
|
|
|
|
|
|
|
host->ShutdownGraphics();
|
2014-06-29 08:09:15 +00:00
|
|
|
|
2017-05-05 13:53:48 +00:00
|
|
|
#if !PPSSPP_PLATFORM(UWP)
|
2012-11-04 09:56:22 +00:00
|
|
|
delete host;
|
2017-04-29 19:02:07 +00:00
|
|
|
host = nullptr;
|
2017-04-05 12:13:25 +00:00
|
|
|
#endif
|
2012-11-01 15:19:01 +00:00
|
|
|
g_Config.Save();
|
2017-04-15 23:30:37 +00:00
|
|
|
|
2018-02-26 15:39:38 +00:00
|
|
|
// Avoid shutting this down when restarting core.
|
2017-04-15 23:30:37 +00:00
|
|
|
if (!restarting)
|
|
|
|
LogManager::Shutdown();
|
2017-03-06 09:51:28 +00:00
|
|
|
|
2013-11-03 00:13:17 +00:00
|
|
|
#ifdef ANDROID_NDK_PROFILER
|
|
|
|
moncleanup();
|
2013-03-29 22:10:40 +00:00
|
|
|
#endif
|
2014-02-09 22:16:08 +00:00
|
|
|
|
|
|
|
ILOG("NativeShutdown called");
|
|
|
|
|
|
|
|
System_SendMessage("finish", "");
|
2017-03-06 09:51:28 +00:00
|
|
|
|
|
|
|
net::Shutdown();
|
|
|
|
|
2018-08-12 22:08:56 +00:00
|
|
|
g_Discord.Shutdown();
|
|
|
|
|
2017-12-31 08:12:34 +00:00
|
|
|
delete logger;
|
|
|
|
logger = nullptr;
|
|
|
|
|
2017-05-31 09:12:11 +00:00
|
|
|
// Previously we did exit() here on Android but that makes it hard to do things like restart on backend change.
|
|
|
|
// I think we handle most globals correctly or correct-enough now.
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
2015-12-17 21:41:50 +00:00
|
|
|
|
2017-08-20 18:03:06 +00:00
|
|
|
void PushNewGpsData(float latitude, float longitude, float altitude, float speed, float bearing, long long time) {
|
|
|
|
GPS::setGpsData(latitude, longitude, altitude, speed, bearing, time);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PushCameraImage(long long length, unsigned char* image) {
|
|
|
|
Camera::pushCameraImage(length, image);
|
|
|
|
}
|