2013-09-07 11:38:37 +00:00
|
|
|
// Copyright (c) 2013- 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
|
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// 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/.
|
|
|
|
|
2013-10-03 20:13:16 +00:00
|
|
|
#include <algorithm>
|
2013-10-11 23:20:10 +00:00
|
|
|
|
2014-09-13 22:14:11 +00:00
|
|
|
#include "base/compat.h"
|
2015-07-23 12:00:02 +00:00
|
|
|
#include "gfx_es2/gpu_features.h"
|
2013-09-07 11:38:37 +00:00
|
|
|
#include "i18n/i18n.h"
|
|
|
|
#include "ui/ui_context.h"
|
|
|
|
#include "ui/view.h"
|
|
|
|
#include "ui/viewgroup.h"
|
|
|
|
#include "ui/ui.h"
|
2015-05-13 20:28:02 +00:00
|
|
|
#include "profiler/profiler.h"
|
2014-07-20 14:32:56 +00:00
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
#include "Common/LogManager.h"
|
2014-07-20 14:32:56 +00:00
|
|
|
#include "Common/CPUDetect.h"
|
|
|
|
|
2013-11-07 12:37:19 +00:00
|
|
|
#include "Core/MemMap.h"
|
2013-09-07 11:38:37 +00:00
|
|
|
#include "Core/Config.h"
|
2013-12-29 23:11:29 +00:00
|
|
|
#include "Core/System.h"
|
2013-11-15 12:11:44 +00:00
|
|
|
#include "Core/CoreParameter.h"
|
2013-09-29 11:39:25 +00:00
|
|
|
#include "Core/MIPS/MIPSTables.h"
|
2016-04-30 23:20:21 +00:00
|
|
|
#include "Core/MIPS/JitCommon/JitBlockCache.h"
|
2013-09-29 11:39:25 +00:00
|
|
|
#include "Core/MIPS/JitCommon/JitCommon.h"
|
2013-11-15 15:49:13 +00:00
|
|
|
#include "GPU/GPUInterface.h"
|
|
|
|
#include "GPU/GPUState.h"
|
2014-07-20 14:32:56 +00:00
|
|
|
#include "UI/MiscScreens.h"
|
|
|
|
#include "UI/DevScreens.h"
|
|
|
|
#include "UI/GameSettingsScreen.h"
|
2013-09-07 18:54:11 +00:00
|
|
|
|
2015-02-02 08:27:27 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
// Want to avoid including the full header here as it includes d3dx.h
|
|
|
|
int GetD3DXVersion();
|
|
|
|
#endif
|
|
|
|
|
2013-10-17 16:23:57 +00:00
|
|
|
static const char *logLevelList[] = {
|
|
|
|
"Notice",
|
|
|
|
"Error",
|
|
|
|
"Warn",
|
|
|
|
"Info",
|
|
|
|
"Debug",
|
|
|
|
"Verb."
|
|
|
|
};
|
|
|
|
|
2013-09-07 18:54:11 +00:00
|
|
|
void DevMenu::CreatePopupContents(UI::ViewGroup *parent) {
|
|
|
|
using namespace UI;
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-09 20:50:51 +00:00
|
|
|
I18NCategory *sy = GetI18NCategory("System");
|
|
|
|
|
2017-03-19 23:36:21 +00:00
|
|
|
ScrollView *scroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, WRAP_CONTENT));
|
2015-12-23 04:46:31 +00:00
|
|
|
LinearLayout *items = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
|
2015-01-05 00:23:03 +00:00
|
|
|
#if !defined(MOBILE_DEVICE)
|
2015-12-23 04:46:31 +00:00
|
|
|
items->Add(new Choice(dev->T("Log View")))->OnClick.Handle(this, &DevMenu::OnLogView);
|
2015-01-05 00:23:03 +00:00
|
|
|
#endif
|
2015-12-23 04:46:31 +00:00
|
|
|
items->Add(new Choice(dev->T("Logging Channels")))->OnClick.Handle(this, &DevMenu::OnLogConfig);
|
|
|
|
items->Add(new Choice(sy->T("Developer Tools")))->OnClick.Handle(this, &DevMenu::OnDeveloperTools);
|
|
|
|
items->Add(new Choice(dev->T("Jit Compare")))->OnClick.Handle(this, &DevMenu::OnJitCompare);
|
|
|
|
items->Add(new Choice(dev->T("Shader Viewer")))->OnClick.Handle(this, &DevMenu::OnShaderView);
|
|
|
|
items->Add(new Choice(dev->T("Toggle Freeze")))->OnClick.Handle(this, &DevMenu::OnFreezeFrame);
|
|
|
|
items->Add(new Choice(dev->T("Dump Frame GPU Commands")))->OnClick.Handle(this, &DevMenu::OnDumpFrame);
|
|
|
|
items->Add(new Choice(dev->T("Toggle Audio Debug")))->OnClick.Handle(this, &DevMenu::OnToggleAudioDebug);
|
2015-05-13 20:28:02 +00:00
|
|
|
#ifdef USE_PROFILER
|
2015-12-23 04:46:31 +00:00
|
|
|
items->Add(new CheckBox(&g_Config.bShowFrameProfiler, dev->T("Frame Profiler"), ""));
|
2015-05-13 20:28:02 +00:00
|
|
|
#endif
|
2015-01-05 00:23:03 +00:00
|
|
|
|
2015-12-23 04:46:31 +00:00
|
|
|
scroll->Add(items);
|
|
|
|
parent->Add(scroll);
|
|
|
|
|
2015-01-05 00:23:03 +00:00
|
|
|
RingbufferLogListener *ring = LogManager::GetInstance()->GetRingbufferListener();
|
|
|
|
if (ring) {
|
2017-03-18 09:47:10 +00:00
|
|
|
ring->SetEnabled(true);
|
2015-01-05 00:23:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-29 11:55:49 +00:00
|
|
|
UI::EventReturn DevMenu::OnToggleAudioDebug(UI::EventParams &e) {
|
|
|
|
g_Config.bShowAudioDebug = !g_Config.bShowAudioDebug;
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-05 00:23:03 +00:00
|
|
|
UI::EventReturn DevMenu::OnLogView(UI::EventParams &e) {
|
|
|
|
UpdateUIState(UISTATE_PAUSEMENU);
|
|
|
|
screenManager()->push(new LogScreen());
|
|
|
|
return UI::EVENT_DONE;
|
2013-09-07 18:54:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn DevMenu::OnLogConfig(UI::EventParams &e) {
|
2014-10-11 16:28:52 +00:00
|
|
|
UpdateUIState(UISTATE_PAUSEMENU);
|
2013-09-07 18:54:11 +00:00
|
|
|
screenManager()->push(new LogConfigScreen());
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn DevMenu::OnDeveloperTools(UI::EventParams &e) {
|
2014-10-11 16:28:52 +00:00
|
|
|
UpdateUIState(UISTATE_PAUSEMENU);
|
2013-09-07 18:54:11 +00:00
|
|
|
screenManager()->push(new DeveloperToolsScreen());
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
UI::EventReturn DevMenu::OnJitCompare(UI::EventParams &e) {
|
2014-10-11 16:28:52 +00:00
|
|
|
UpdateUIState(UISTATE_PAUSEMENU);
|
2013-09-29 11:39:25 +00:00
|
|
|
screenManager()->push(new JitCompareScreen());
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
UI::EventReturn DevMenu::OnShaderView(UI::EventParams &e) {
|
|
|
|
UpdateUIState(UISTATE_PAUSEMENU);
|
|
|
|
screenManager()->push(new ShaderListScreen());
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2013-11-15 12:11:44 +00:00
|
|
|
UI::EventReturn DevMenu::OnFreezeFrame(UI::EventParams &e) {
|
|
|
|
if (PSP_CoreParameter().frozen) {
|
|
|
|
PSP_CoreParameter().frozen = false;
|
|
|
|
} else {
|
|
|
|
PSP_CoreParameter().freezeNext = true;
|
|
|
|
}
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-11-15 15:49:13 +00:00
|
|
|
UI::EventReturn DevMenu::OnDumpFrame(UI::EventParams &e) {
|
|
|
|
gpu->DumpNextFrame();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-09-07 20:02:55 +00:00
|
|
|
void DevMenu::dialogFinished(const Screen *dialog, DialogResult result) {
|
2015-01-29 11:55:49 +00:00
|
|
|
UpdateUIState(UISTATE_INGAME);
|
2013-09-07 20:02:55 +00:00
|
|
|
// Close when a subscreen got closed.
|
|
|
|
// TODO: a bug in screenmanager causes this not to work here.
|
2017-03-20 00:43:03 +00:00
|
|
|
// TriggerFinish(DR_OK);
|
2013-09-07 20:02:55 +00:00
|
|
|
}
|
|
|
|
|
2015-01-05 00:23:03 +00:00
|
|
|
void LogScreen::UpdateLog() {
|
|
|
|
using namespace UI;
|
|
|
|
RingbufferLogListener *ring = LogManager::GetInstance()->GetRingbufferListener();
|
|
|
|
if (!ring)
|
|
|
|
return;
|
|
|
|
vert_->Clear();
|
|
|
|
for (int i = ring->GetCount() - 1; i >= 0; i--) {
|
|
|
|
TextView *v = vert_->Add(new TextView(ring->TextAt(i), FLAG_DYNAMIC_ASCII, false));
|
|
|
|
uint32_t color = 0xFFFFFF;
|
|
|
|
switch (ring->LevelAt(i)) {
|
|
|
|
case LogTypes::LDEBUG: color = 0xE0E0E0; break;
|
|
|
|
case LogTypes::LWARNING: color = 0x50FFFF; break;
|
|
|
|
case LogTypes::LERROR: color = 0x5050FF; break;
|
|
|
|
case LogTypes::LNOTICE: color = 0x30FF30; break;
|
|
|
|
case LogTypes::LINFO: color = 0xFFFFFF; break;
|
2015-01-10 00:52:20 +00:00
|
|
|
case LogTypes::LVERBOSE: color = 0xC0C0C0; break;
|
2015-01-05 00:23:03 +00:00
|
|
|
}
|
|
|
|
v->SetTextColor(0xFF000000 | color);
|
|
|
|
}
|
|
|
|
toBottom_ = true;
|
|
|
|
}
|
|
|
|
|
2017-03-15 05:01:18 +00:00
|
|
|
void LogScreen::update() {
|
|
|
|
UIDialogScreenWithBackground::update();
|
2015-01-05 00:23:03 +00:00
|
|
|
if (toBottom_) {
|
|
|
|
toBottom_ = false;
|
|
|
|
scroll_->ScrollToBottom();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogScreen::CreateViews() {
|
|
|
|
using namespace UI;
|
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
|
|
|
|
|
|
|
LinearLayout *outer = new LinearLayout(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, WRAP_CONTENT));
|
|
|
|
root_ = outer;
|
|
|
|
|
|
|
|
scroll_ = outer->Add(new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(1.0)));
|
|
|
|
LinearLayout *bottom = outer->Add(new LinearLayout(ORIENT_HORIZONTAL, new LayoutParams(FILL_PARENT, WRAP_CONTENT)));
|
|
|
|
bottom->Add(new Button(di->T("Back")))->OnClick.Handle<UIScreen>(this, &UIScreen::OnBack);
|
|
|
|
cmdLine_ = bottom->Add(new TextEdit("", "Command Line", new LinearLayoutParams(1.0)));
|
|
|
|
cmdLine_->OnEnter.Handle(this, &LogScreen::OnSubmit);
|
|
|
|
bottom->Add(new Button(di->T("Submit")))->OnClick.Handle(this, &LogScreen::OnSubmit);
|
|
|
|
|
|
|
|
vert_ = scroll_->Add(new LinearLayout(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, WRAP_CONTENT)));
|
|
|
|
vert_->SetSpacing(0);
|
|
|
|
|
|
|
|
UpdateLog();
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn LogScreen::OnSubmit(UI::EventParams &e) {
|
|
|
|
std::string cmd = cmdLine_->GetText();
|
|
|
|
|
|
|
|
// TODO: Can add all sorts of fun stuff here that we can't be bothered writing proper UI for, like various memdumps etc.
|
|
|
|
|
2017-03-06 12:10:23 +00:00
|
|
|
NOTICE_LOG(SYSTEM, "Submitted: %s", cmd.c_str());
|
2015-01-05 00:23:03 +00:00
|
|
|
|
|
|
|
UpdateLog();
|
|
|
|
cmdLine_->SetText("");
|
|
|
|
cmdLine_->SetFocus();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
2013-09-07 20:02:55 +00:00
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
void LogConfigScreen::CreateViews() {
|
|
|
|
using namespace UI;
|
|
|
|
|
2014-12-09 20:50:51 +00:00
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2013-09-07 11:38:37 +00:00
|
|
|
|
|
|
|
root_ = new ScrollView(ORIENT_VERTICAL);
|
2013-09-07 18:54:11 +00:00
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
LinearLayout *vert = root_->Add(new LinearLayout(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, WRAP_CONTENT)));
|
|
|
|
vert->SetSpacing(0);
|
|
|
|
|
2013-09-07 18:54:11 +00:00
|
|
|
LinearLayout *topbar = new LinearLayout(ORIENT_HORIZONTAL);
|
2014-12-09 20:50:51 +00:00
|
|
|
topbar->Add(new Choice(di->T("Back")))->OnClick.Handle<UIScreen>(this, &UIScreen::OnBack);
|
|
|
|
topbar->Add(new Choice(di->T("Toggle All")))->OnClick.Handle(this, &LogConfigScreen::OnToggleAll);
|
2017-03-06 12:50:22 +00:00
|
|
|
topbar->Add(new Choice(di->T("Enable All")))->OnClick.Handle(this, &LogConfigScreen::OnEnableAll);
|
|
|
|
topbar->Add(new Choice(di->T("Disable All")))->OnClick.Handle(this, &LogConfigScreen::OnDisableAll);
|
2015-06-30 03:28:31 +00:00
|
|
|
topbar->Add(new Choice(dev->T("Log Level")))->OnClick.Handle(this, &LogConfigScreen::OnLogLevel);
|
2013-09-07 18:54:11 +00:00
|
|
|
|
|
|
|
vert->Add(topbar);
|
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
vert->Add(new ItemHeader(dev->T("Logging Channels")));
|
2013-09-07 11:38:37 +00:00
|
|
|
|
|
|
|
LogManager *logMan = LogManager::GetInstance();
|
|
|
|
|
2013-09-07 20:02:55 +00:00
|
|
|
int cellSize = 400;
|
|
|
|
|
|
|
|
UI::GridLayoutSettings gridsettings(cellSize, 64, 5);
|
|
|
|
gridsettings.fillCells = true;
|
|
|
|
GridLayout *grid = vert->Add(new GridLayout(gridsettings, new LayoutParams(FILL_PARENT, WRAP_CONTENT)));
|
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
for (int i = 0; i < LogManager::GetNumChannels(); i++) {
|
|
|
|
LogTypes::LOG_TYPE type = (LogTypes::LOG_TYPE)i;
|
|
|
|
LogChannel *chan = logMan->GetLogChannel(type);
|
2013-09-07 20:02:55 +00:00
|
|
|
LinearLayout *row = new LinearLayout(ORIENT_HORIZONTAL, new LinearLayoutParams(cellSize - 50, WRAP_CONTENT));
|
|
|
|
row->SetSpacing(0);
|
2017-03-18 09:47:10 +00:00
|
|
|
row->Add(new CheckBox(&chan->enabled, "", "", new LinearLayoutParams(50, WRAP_CONTENT)));
|
|
|
|
row->Add(new PopupMultiChoice((int *)&chan->level, chan->m_shortName, logLevelList, 1, 6, 0, screenManager(), new LinearLayoutParams(1.0)));
|
2013-09-07 20:02:55 +00:00
|
|
|
grid->Add(row);
|
2013-09-07 11:38:37 +00:00
|
|
|
}
|
2013-09-07 18:54:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn LogConfigScreen::OnToggleAll(UI::EventParams &e) {
|
|
|
|
LogManager *logMan = LogManager::GetInstance();
|
|
|
|
for (int i = 0; i < LogManager::GetNumChannels(); i++) {
|
2017-03-06 12:50:22 +00:00
|
|
|
LogChannel *chan = logMan->GetLogChannel((LogTypes::LOG_TYPE)i);
|
2017-03-18 09:47:10 +00:00
|
|
|
chan->enabled = !chan->enabled;
|
2013-09-07 18:54:11 +00:00
|
|
|
}
|
2017-03-06 12:50:22 +00:00
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
2013-09-07 11:38:37 +00:00
|
|
|
|
2017-03-06 12:50:22 +00:00
|
|
|
UI::EventReturn LogConfigScreen::OnEnableAll(UI::EventParams &e) {
|
|
|
|
LogManager *logMan = LogManager::GetInstance();
|
|
|
|
for (int i = 0; i < LogManager::GetNumChannels(); i++) {
|
|
|
|
LogChannel *chan = logMan->GetLogChannel((LogTypes::LOG_TYPE)i);
|
2017-03-18 09:47:10 +00:00
|
|
|
chan->enabled = true;
|
2017-03-06 12:50:22 +00:00
|
|
|
}
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn LogConfigScreen::OnDisableAll(UI::EventParams &e) {
|
|
|
|
LogManager *logMan = LogManager::GetInstance();
|
|
|
|
for (int i = 0; i < LogManager::GetNumChannels(); i++) {
|
|
|
|
LogChannel *chan = logMan->GetLogChannel((LogTypes::LOG_TYPE)i);
|
2017-03-18 09:47:10 +00:00
|
|
|
chan->enabled = false;
|
2017-03-06 12:50:22 +00:00
|
|
|
}
|
2013-09-07 18:54:11 +00:00
|
|
|
return UI::EVENT_DONE;
|
2013-09-07 11:38:37 +00:00
|
|
|
}
|
|
|
|
|
2013-10-17 16:23:57 +00:00
|
|
|
UI::EventReturn LogConfigScreen::OnLogLevelChange(UI::EventParams &e) {
|
|
|
|
RecreateViews();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn LogConfigScreen::OnLogLevel(UI::EventParams &e) {
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-10 22:01:09 +00:00
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
auto logLevelScreen = new LogLevelScreen(dev->T("Log Level"));
|
2013-10-17 16:23:57 +00:00
|
|
|
logLevelScreen->OnChoice.Handle(this, &LogConfigScreen::OnLogLevelChange);
|
2017-03-22 01:27:57 +00:00
|
|
|
if (e.v)
|
|
|
|
logLevelScreen->SetPopupOrigin(e.v);
|
2013-10-17 16:23:57 +00:00
|
|
|
screenManager()->push(logLevelScreen);
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
LogLevelScreen::LogLevelScreen(const std::string &title) : ListPopupScreen(title) {
|
|
|
|
int NUMLOGLEVEL = 6;
|
|
|
|
std::vector<std::string> list;
|
2017-03-22 01:27:57 +00:00
|
|
|
for (int i = 0; i < NUMLOGLEVEL; ++i) {
|
2013-10-17 16:23:57 +00:00
|
|
|
list.push_back(logLevelList[i]);
|
|
|
|
}
|
|
|
|
adaptor_ = UI::StringVectorListAdaptor(list, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LogLevelScreen::OnCompleted(DialogResult result) {
|
|
|
|
if (result != DR_OK)
|
|
|
|
return;
|
|
|
|
int selected = listView_->GetSelected();
|
|
|
|
LogManager *logMan = LogManager::GetInstance();
|
|
|
|
|
|
|
|
for (int i = 0; i < LogManager::GetNumChannels(); ++i) {
|
|
|
|
LogTypes::LOG_TYPE type = (LogTypes::LOG_TYPE)i;
|
|
|
|
LogChannel *chan = logMan->GetLogChannel(type);
|
2017-03-18 09:47:10 +00:00
|
|
|
if (chan->enabled)
|
|
|
|
chan->level = (LogTypes::LOG_LEVELS)(selected + 1);
|
2013-10-17 16:23:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-20 14:32:56 +00:00
|
|
|
const char *GetCompilerABI() {
|
2016-10-12 15:32:52 +00:00
|
|
|
#if PPSSPP_ARCH(ARMV7)
|
2014-07-20 14:32:56 +00:00
|
|
|
return "armeabi-v7a";
|
2016-10-12 15:32:52 +00:00
|
|
|
#elif PPSSPP_ARCH(ARM)
|
2014-07-20 15:34:39 +00:00
|
|
|
return "armeabi";
|
2016-10-12 15:32:52 +00:00
|
|
|
#elif PPSSPP_ARCH(ARM64)
|
2015-03-05 22:31:03 +00:00
|
|
|
return "arm64";
|
2016-10-12 15:32:52 +00:00
|
|
|
#elif PPSSPP_ARCH(X86)
|
2014-07-20 14:32:56 +00:00
|
|
|
return "x86";
|
2016-10-12 15:32:52 +00:00
|
|
|
#elif PPSSPP_ARCH(AMD64)
|
2014-07-20 14:32:56 +00:00
|
|
|
return "x86-64";
|
|
|
|
#else
|
|
|
|
return "other";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
void SystemInfoScreen::CreateViews() {
|
2016-12-25 17:18:19 +00:00
|
|
|
using namespace Draw;
|
2016-12-26 16:03:01 +00:00
|
|
|
using namespace UI;
|
|
|
|
|
2013-09-07 11:38:37 +00:00
|
|
|
// NOTE: Do not translate this section. It will change a lot and will be impossible to keep up.
|
2015-07-01 21:26:55 +00:00
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
2013-10-11 23:20:10 +00:00
|
|
|
root_ = new AnchorLayout(new LayoutParams(FILL_PARENT, FILL_PARENT));
|
|
|
|
|
|
|
|
ViewGroup *leftColumn = new AnchorLayout(new LinearLayoutParams(1.0f));
|
|
|
|
root_->Add(leftColumn);
|
|
|
|
|
2015-07-04 15:01:32 +00:00
|
|
|
AddStandardBack(root_);
|
2013-10-11 23:20:10 +00:00
|
|
|
|
|
|
|
TabHolder *tabHolder = new TabHolder(ORIENT_VERTICAL, 225, new AnchorLayoutParams(10, 0, 10, 0, false));
|
2016-01-23 06:38:21 +00:00
|
|
|
tabHolder->SetTag("DevSystemInfo");
|
2013-10-11 23:20:10 +00:00
|
|
|
|
|
|
|
root_->Add(tabHolder);
|
|
|
|
ViewGroup *deviceSpecsScroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, FILL_PARENT));
|
2016-01-23 06:52:13 +00:00
|
|
|
deviceSpecsScroll->SetTag("DevSystemInfoDeviceSpecs");
|
2013-10-11 23:20:10 +00:00
|
|
|
LinearLayout *deviceSpecs = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
deviceSpecs->SetSpacing(0);
|
|
|
|
deviceSpecsScroll->Add(deviceSpecs);
|
|
|
|
tabHolder->AddTab("Device Info", deviceSpecsScroll);
|
|
|
|
|
|
|
|
deviceSpecs->Add(new ItemHeader("System Information"));
|
|
|
|
deviceSpecs->Add(new InfoItem("Name", System_GetProperty(SYSPROP_NAME)));
|
|
|
|
deviceSpecs->Add(new InfoItem("Lang/Region", System_GetProperty(SYSPROP_LANGREGION)));
|
2014-07-20 14:32:56 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("ABI", GetCompilerABI()));
|
2017-05-21 15:24:40 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
if (IsDebuggerPresent()) {
|
|
|
|
deviceSpecs->Add(new InfoItem("Debugger Present", "Yes"));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-10-11 23:20:10 +00:00
|
|
|
deviceSpecs->Add(new ItemHeader("CPU Information"));
|
|
|
|
deviceSpecs->Add(new InfoItem("Name", cpu_info.brand_string));
|
2015-06-27 04:39:48 +00:00
|
|
|
#if defined(ARM) || defined(ARM64) || defined(MIPS)
|
2013-10-15 05:28:19 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("Cores", StringFromInt(cpu_info.num_cores)));
|
|
|
|
#else
|
2013-10-16 20:39:53 +00:00
|
|
|
int totalThreads = cpu_info.num_cores * cpu_info.logical_cpu_count;
|
|
|
|
std::string cores = StringFromFormat("%d (%d per core, %d cores)", totalThreads, cpu_info.logical_cpu_count, cpu_info.num_cores);
|
|
|
|
deviceSpecs->Add(new InfoItem("Threads", cores));
|
2013-10-15 05:28:19 +00:00
|
|
|
#endif
|
2013-10-11 23:20:10 +00:00
|
|
|
deviceSpecs->Add(new ItemHeader("GPU Information"));
|
2014-08-22 18:48:35 +00:00
|
|
|
|
2017-01-30 13:33:38 +00:00
|
|
|
DrawContext *draw = screenManager()->getDrawContext();
|
2014-08-22 18:48:35 +00:00
|
|
|
|
2017-02-08 13:55:03 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("3D API", std::string(draw->GetInfoString(InfoField::APINAME))));
|
2017-03-16 08:48:10 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("Vendor", std::string(draw->GetInfoString(InfoField::VENDORSTRING))));
|
|
|
|
std::string vendor = draw->GetInfoString(InfoField::VENDOR);
|
|
|
|
if (vendor.size())
|
|
|
|
deviceSpecs->Add(new InfoItem("Vendor (detected)", vendor));
|
|
|
|
deviceSpecs->Add(new InfoItem("Driver", draw->GetInfoString(InfoField::DRIVER)));
|
2014-07-31 05:21:37 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
deviceSpecs->Add(new InfoItem("Driver Version", System_GetProperty(SYSPROP_GPUDRIVER_VERSION)));
|
2017-02-24 23:25:46 +00:00
|
|
|
#if !PPSSPP_PLATFORM(UWP)
|
2016-01-06 06:37:28 +00:00
|
|
|
if (GetGPUBackend() == GPUBackend::DIRECT3D9) {
|
2015-02-02 08:27:27 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("D3DX Version", StringFromFormat("%d", GetD3DXVersion())));
|
|
|
|
}
|
2017-02-24 23:25:46 +00:00
|
|
|
#endif
|
2014-07-31 05:21:37 +00:00
|
|
|
#endif
|
2017-01-26 08:20:05 +00:00
|
|
|
deviceSpecs->Add(new ItemHeader("OS Information"));
|
|
|
|
deviceSpecs->Add(new InfoItem("Memory Page Size", StringFromFormat("%d bytes", GetMemoryProtectPageSize())));
|
|
|
|
deviceSpecs->Add(new InfoItem("RW/RX exclusive: ", PlatformIsWXExclusive() ? "Yes" : "No"));
|
2015-01-11 10:40:53 +00:00
|
|
|
|
2017-02-28 00:47:13 +00:00
|
|
|
const char *build = "Release";
|
|
|
|
#ifdef _DEBUG
|
|
|
|
build = "Debug";
|
|
|
|
#endif
|
|
|
|
deviceSpecs->Add(new InfoItem("PPSSPP build: ", build));
|
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#ifdef __ANDROID__
|
2015-01-11 10:40:53 +00:00
|
|
|
deviceSpecs->Add(new ItemHeader("Audio Information"));
|
|
|
|
deviceSpecs->Add(new InfoItem("Sample rate", StringFromFormat("%d Hz", System_GetPropertyInt(SYSPROP_AUDIO_SAMPLE_RATE))));
|
|
|
|
deviceSpecs->Add(new InfoItem("Frames per buffer", StringFromFormat("%d", System_GetPropertyInt(SYSPROP_AUDIO_FRAMES_PER_BUFFER))));
|
|
|
|
deviceSpecs->Add(new InfoItem("Optimal sample rate", StringFromFormat("%d Hz", System_GetPropertyInt(SYSPROP_AUDIO_OPTIMAL_SAMPLE_RATE))));
|
|
|
|
deviceSpecs->Add(new InfoItem("Optimal frames per buffer", StringFromFormat("%d", System_GetPropertyInt(SYSPROP_AUDIO_OPTIMAL_FRAMES_PER_BUFFER))));
|
2015-01-13 23:45:12 +00:00
|
|
|
|
|
|
|
deviceSpecs->Add(new ItemHeader("Display Information"));
|
|
|
|
deviceSpecs->Add(new InfoItem("Native Resolution", StringFromFormat("%dx%d",
|
|
|
|
System_GetPropertyInt(SYSPROP_DISPLAY_XRES),
|
|
|
|
System_GetPropertyInt(SYSPROP_DISPLAY_YRES))));
|
|
|
|
deviceSpecs->Add(new InfoItem("Refresh rate", StringFromFormat("%0.3f Hz", (float)System_GetPropertyInt(SYSPROP_DISPLAY_REFRESH_RATE) / 1000.0f)));
|
2015-01-11 10:40:53 +00:00
|
|
|
#endif
|
|
|
|
|
2015-01-13 23:45:12 +00:00
|
|
|
|
2014-08-22 18:48:35 +00:00
|
|
|
deviceSpecs->Add(new ItemHeader("Version Information"));
|
2015-07-23 12:00:02 +00:00
|
|
|
std::string apiVersion;
|
2016-01-06 06:37:28 +00:00
|
|
|
if (GetGPUBackend() == GPUBackend::OPENGL) {
|
2015-09-05 23:45:17 +00:00
|
|
|
if (gl_extensions.IsGLES) {
|
|
|
|
apiVersion = StringFromFormat("v%d.%d.%d ES", gl_extensions.ver[0], gl_extensions.ver[1], gl_extensions.ver[2]);
|
|
|
|
} else {
|
|
|
|
apiVersion = StringFromFormat("v%d.%d.%d", gl_extensions.ver[0], gl_extensions.ver[1], gl_extensions.ver[2]);
|
|
|
|
}
|
2015-07-23 12:00:02 +00:00
|
|
|
} else {
|
2017-01-30 13:33:38 +00:00
|
|
|
apiVersion = draw->GetInfoString(InfoField::APIVERSION);
|
2015-07-23 12:00:02 +00:00
|
|
|
if (apiVersion.size() > 30)
|
|
|
|
apiVersion.resize(30);
|
|
|
|
}
|
2014-08-22 18:48:35 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("API Version", apiVersion));
|
2017-01-30 13:33:38 +00:00
|
|
|
deviceSpecs->Add(new InfoItem("Shading Language", draw->GetInfoString(InfoField::SHADELANGVERSION)));
|
2013-11-14 15:17:22 +00:00
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#ifdef __ANDROID__
|
2015-01-01 12:35:31 +00:00
|
|
|
std::string moga = System_GetProperty(SYSPROP_MOGA_VERSION);
|
|
|
|
if (moga.empty()) {
|
|
|
|
moga = "(none detected)";
|
|
|
|
}
|
|
|
|
deviceSpecs->Add(new InfoItem("Moga", moga));
|
|
|
|
#endif
|
|
|
|
|
2016-10-12 09:13:16 +00:00
|
|
|
#ifdef __ANDROID__
|
2014-07-20 23:19:42 +00:00
|
|
|
char temp[256];
|
|
|
|
sprintf(temp, "%dx%d", System_GetPropertyInt(SYSPROP_DISPLAY_XRES), System_GetPropertyInt(SYSPROP_DISPLAY_YRES));
|
|
|
|
deviceSpecs->Add(new InfoItem("Display resolution", temp));
|
|
|
|
#endif
|
|
|
|
|
2017-03-16 11:33:23 +00:00
|
|
|
ViewGroup *buildConfigScroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, FILL_PARENT));
|
|
|
|
buildConfigScroll->SetTag("DevSystemInfoBuildConfig");
|
|
|
|
LinearLayout *buildConfig = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
buildConfig->SetSpacing(0);
|
|
|
|
buildConfigScroll->Add(buildConfig);
|
|
|
|
tabHolder->AddTab("Build Config", buildConfigScroll);
|
|
|
|
|
|
|
|
buildConfig->Add(new ItemHeader("Build Configuration"));
|
2017-03-17 10:55:14 +00:00
|
|
|
#ifdef JENKINS
|
|
|
|
buildConfig->Add(new InfoItem("Built by", "Jenkins"));
|
|
|
|
#endif
|
2017-03-16 11:33:23 +00:00
|
|
|
#ifdef _DEBUG
|
|
|
|
buildConfig->Add(new InfoItem("_DEBUG", ""));
|
|
|
|
#else
|
|
|
|
buildConfig->Add(new InfoItem("NDEBUG", ""));
|
|
|
|
#endif
|
|
|
|
#ifdef USING_GLES2
|
|
|
|
buildConfig->Add(new InfoItem("USING_GLES2", ""));
|
|
|
|
#endif
|
|
|
|
#ifdef MOBILE_DEVICE
|
|
|
|
buildConfig->Add(new InfoItem("MOBILE_DEVICE", ""));
|
|
|
|
#endif
|
2017-04-30 00:35:12 +00:00
|
|
|
if (System_GetPropertyBool(SYSPROP_APP_GOLD)) {
|
2017-04-05 14:21:08 +00:00
|
|
|
buildConfig->Add(new InfoItem("GOLD", ""));
|
|
|
|
}
|
2017-03-16 11:33:23 +00:00
|
|
|
|
2013-11-14 15:17:22 +00:00
|
|
|
ViewGroup *cpuExtensionsScroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, FILL_PARENT));
|
2016-01-23 06:52:13 +00:00
|
|
|
cpuExtensionsScroll->SetTag("DevSystemInfoCPUExt");
|
2013-11-14 15:17:22 +00:00
|
|
|
LinearLayout *cpuExtensions = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
cpuExtensions->SetSpacing(0);
|
|
|
|
cpuExtensionsScroll->Add(cpuExtensions);
|
|
|
|
|
|
|
|
tabHolder->AddTab("CPU Extensions", cpuExtensionsScroll);
|
|
|
|
|
|
|
|
cpuExtensions->Add(new ItemHeader("CPU Extensions"));
|
|
|
|
std::vector<std::string> exts;
|
|
|
|
SplitString(cpu_info.Summarize(), ',', exts);
|
|
|
|
for (size_t i = 2; i < exts.size(); i++) {
|
2015-06-29 01:15:00 +00:00
|
|
|
cpuExtensions->Add(new TextView(exts[i]))->SetFocusable(true);
|
2013-11-14 15:17:22 +00:00
|
|
|
}
|
2015-07-21 18:48:10 +00:00
|
|
|
|
2013-10-11 23:20:10 +00:00
|
|
|
ViewGroup *oglExtensionsScroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, FILL_PARENT));
|
2016-01-23 06:52:13 +00:00
|
|
|
oglExtensionsScroll->SetTag("DevSystemInfoOGLExt");
|
2013-10-11 23:20:10 +00:00
|
|
|
LinearLayout *oglExtensions = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
oglExtensions->SetSpacing(0);
|
|
|
|
oglExtensionsScroll->Add(oglExtensions);
|
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
if (g_Config.iGPUBackend == GPU_BACKEND_OPENGL) {
|
|
|
|
tabHolder->AddTab("OGL Extensions", oglExtensionsScroll);
|
2013-10-11 23:20:10 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
if (!gl_extensions.IsGLES) {
|
|
|
|
oglExtensions->Add(new ItemHeader("OpenGL Extensions"));
|
|
|
|
} else if (gl_extensions.GLES3) {
|
|
|
|
oglExtensions->Add(new ItemHeader("OpenGL ES 3.0 Extensions"));
|
|
|
|
} else {
|
|
|
|
oglExtensions->Add(new ItemHeader("OpenGL ES 2.0 Extensions"));
|
|
|
|
}
|
|
|
|
exts.clear();
|
|
|
|
SplitString(g_all_gl_extensions, ' ', exts);
|
|
|
|
std::sort(exts.begin(), exts.end());
|
|
|
|
for (size_t i = 0; i < exts.size(); i++) {
|
|
|
|
oglExtensions->Add(new TextView(exts[i]))->SetFocusable(true);
|
|
|
|
}
|
2013-09-07 11:38:37 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
exts.clear();
|
|
|
|
SplitString(g_all_egl_extensions, ' ', exts);
|
|
|
|
std::sort(exts.begin(), exts.end());
|
2014-05-06 04:25:50 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
// If there aren't any EGL extensions, no need to show the tab.
|
|
|
|
if (exts.size() > 0) {
|
|
|
|
ViewGroup *eglExtensionsScroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, FILL_PARENT));
|
|
|
|
eglExtensionsScroll->SetTag("DevSystemInfoEGLExt");
|
|
|
|
LinearLayout *eglExtensions = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
eglExtensions->SetSpacing(0);
|
|
|
|
eglExtensionsScroll->Add(eglExtensions);
|
2014-05-06 04:25:50 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
tabHolder->AddTab("EGL Extensions", eglExtensionsScroll);
|
2014-05-06 04:25:50 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
eglExtensions->Add(new ItemHeader("EGL Extensions"));
|
2014-05-06 04:25:50 +00:00
|
|
|
|
2016-01-03 13:00:05 +00:00
|
|
|
for (size_t i = 0; i < exts.size(); i++) {
|
|
|
|
eglExtensions->Add(new TextView(exts[i]))->SetFocusable(true);
|
|
|
|
}
|
2014-05-06 04:25:50 +00:00
|
|
|
}
|
2016-01-03 13:00:05 +00:00
|
|
|
} else if (g_Config.iGPUBackend == GPU_BACKEND_VULKAN) {
|
|
|
|
tabHolder->AddTab("Vulkan Features", oglExtensionsScroll);
|
|
|
|
|
|
|
|
oglExtensions->Add(new ItemHeader("Vulkan Features"));
|
2017-01-30 13:33:38 +00:00
|
|
|
std::vector<std::string> features = draw->GetFeatureList();
|
2016-01-03 13:00:05 +00:00
|
|
|
for (auto &feature : features) {
|
|
|
|
oglExtensions->Add(new TextView(feature))->SetFocusable(true);
|
|
|
|
}
|
2013-09-07 11:38:37 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-29 11:39:25 +00:00
|
|
|
|
2013-11-29 06:35:03 +00:00
|
|
|
void AddressPromptScreen::CreatePopupContents(UI::ViewGroup *parent) {
|
2013-11-29 07:02:04 +00:00
|
|
|
using namespace UI;
|
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-10 22:01:09 +00:00
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
addrView_ = new TextView(dev->T("Enter address"), ALIGN_HCENTER, false);
|
2013-11-29 06:35:03 +00:00
|
|
|
parent->Add(addrView_);
|
2013-11-29 07:02:04 +00:00
|
|
|
|
|
|
|
ViewGroup *grid = new GridLayout(GridLayoutSettings(60, 40));
|
|
|
|
parent->Add(grid);
|
|
|
|
|
|
|
|
for (int i = 0; i < 16; ++i) {
|
|
|
|
char temp[16];
|
|
|
|
snprintf(temp, 16, " %X ", i);
|
|
|
|
buttons_[i] = new Button(temp);
|
|
|
|
grid->Add(buttons_[i])->OnClick.Handle(this, &AddressPromptScreen::OnDigitButton);
|
|
|
|
}
|
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
parent->Add(new Button(dev->T("Backspace")))->OnClick.Handle(this, &AddressPromptScreen::OnBackspace);
|
2013-11-29 06:35:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void AddressPromptScreen::OnCompleted(DialogResult result) {
|
|
|
|
if (result == DR_OK) {
|
2017-03-22 01:34:52 +00:00
|
|
|
UI::EventParams e{};
|
2013-11-29 06:35:03 +00:00
|
|
|
e.v = root_;
|
|
|
|
e.a = addr_;
|
|
|
|
OnChoice.Trigger(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-29 07:02:04 +00:00
|
|
|
UI::EventReturn AddressPromptScreen::OnDigitButton(UI::EventParams &e) {
|
|
|
|
for (int i = 0; i < 16; ++i) {
|
|
|
|
if (buttons_[i] == e.v) {
|
|
|
|
AddDigit(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn AddressPromptScreen::OnBackspace(UI::EventParams &e) {
|
|
|
|
BackspaceDigit();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddressPromptScreen::AddDigit(int n) {
|
|
|
|
if ((addr_ & 0xF0000000) == 0) {
|
|
|
|
addr_ = addr_ * 16 + n;
|
|
|
|
}
|
|
|
|
UpdatePreviewDigits();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddressPromptScreen::BackspaceDigit() {
|
|
|
|
addr_ /= 16;
|
|
|
|
UpdatePreviewDigits();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddressPromptScreen::UpdatePreviewDigits() {
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-10 22:01:09 +00:00
|
|
|
|
2013-11-29 07:02:04 +00:00
|
|
|
if (addr_ != 0) {
|
|
|
|
char temp[32];
|
|
|
|
snprintf(temp, 32, "%8X", addr_);
|
|
|
|
addrView_->SetText(temp);
|
|
|
|
} else {
|
2015-06-30 03:28:31 +00:00
|
|
|
addrView_->SetText(dev->T("Enter address"));
|
2013-11-29 07:02:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-15 11:04:59 +00:00
|
|
|
bool AddressPromptScreen::key(const KeyInput &key) {
|
2013-11-29 06:35:03 +00:00
|
|
|
if (key.flags & KEY_DOWN) {
|
|
|
|
if (key.keyCode >= NKCODE_0 && key.keyCode <= NKCODE_9) {
|
2013-11-29 07:02:04 +00:00
|
|
|
AddDigit(key.keyCode - NKCODE_0);
|
2013-11-29 06:35:03 +00:00
|
|
|
} else if (key.keyCode >= NKCODE_A && key.keyCode <= NKCODE_F) {
|
2013-11-29 07:02:04 +00:00
|
|
|
AddDigit(10 + key.keyCode - NKCODE_A);
|
2013-11-29 06:35:03 +00:00
|
|
|
// NKCODE_DEL is backspace.
|
|
|
|
} else if (key.keyCode == NKCODE_DEL) {
|
2013-11-29 07:02:04 +00:00
|
|
|
BackspaceDigit();
|
2013-11-29 06:35:03 +00:00
|
|
|
} else if (key.keyCode == NKCODE_ENTER) {
|
2017-03-20 00:43:03 +00:00
|
|
|
TriggerFinish(DR_OK);
|
2013-11-29 06:35:03 +00:00
|
|
|
} else {
|
2014-06-15 11:04:59 +00:00
|
|
|
return UIDialogScreen::key(key);
|
2013-11-29 06:35:03 +00:00
|
|
|
}
|
|
|
|
} else {
|
2014-06-15 11:04:59 +00:00
|
|
|
return UIDialogScreen::key(key);
|
2013-11-29 06:35:03 +00:00
|
|
|
}
|
2014-06-15 11:04:59 +00:00
|
|
|
return true;
|
2013-11-29 06:35:03 +00:00
|
|
|
}
|
2013-09-29 11:39:25 +00:00
|
|
|
|
|
|
|
// Three panes: Block chooser, MIPS view, ARM/x86 view
|
|
|
|
void JitCompareScreen::CreateViews() {
|
2015-07-01 21:26:55 +00:00
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2013-09-29 11:39:25 +00:00
|
|
|
|
|
|
|
using namespace UI;
|
|
|
|
|
|
|
|
root_ = new LinearLayout(ORIENT_HORIZONTAL);
|
|
|
|
|
|
|
|
ScrollView *leftColumnScroll = root_->Add(new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(1.0f)));
|
|
|
|
LinearLayout *leftColumn = leftColumnScroll->Add(new LinearLayout(ORIENT_VERTICAL));
|
|
|
|
|
|
|
|
ScrollView *midColumnScroll = root_->Add(new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(2.0f)));
|
|
|
|
LinearLayout *midColumn = midColumnScroll->Add(new LinearLayout(ORIENT_VERTICAL));
|
2016-01-23 06:52:13 +00:00
|
|
|
midColumn->SetTag("JitCompareLeftDisasm");
|
2013-09-29 11:39:25 +00:00
|
|
|
leftDisasm_ = midColumn->Add(new LinearLayout(ORIENT_VERTICAL));
|
|
|
|
leftDisasm_->SetSpacing(0.0f);
|
|
|
|
|
|
|
|
ScrollView *rightColumnScroll = root_->Add(new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(2.0f)));
|
2016-01-23 06:52:13 +00:00
|
|
|
rightColumnScroll->SetTag("JitCompareRightDisasm");
|
2013-09-29 11:39:25 +00:00
|
|
|
LinearLayout *rightColumn = rightColumnScroll->Add(new LinearLayout(ORIENT_VERTICAL));
|
|
|
|
rightDisasm_ = rightColumn->Add(new LinearLayout(ORIENT_VERTICAL));
|
|
|
|
rightDisasm_->SetSpacing(0.0f);
|
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
leftColumn->Add(new Choice(dev->T("Current")))->OnClick.Handle(this, &JitCompareScreen::OnCurrentBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("By Address")))->OnClick.Handle(this, &JitCompareScreen::OnSelectBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("Prev")))->OnClick.Handle(this, &JitCompareScreen::OnPrevBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("Next")))->OnClick.Handle(this, &JitCompareScreen::OnNextBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("Random")))->OnClick.Handle(this, &JitCompareScreen::OnRandomBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("FPU")))->OnClick.Handle(this, &JitCompareScreen::OnRandomFPUBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("VFPU")))->OnClick.Handle(this, &JitCompareScreen::OnRandomVFPUBlock);
|
|
|
|
leftColumn->Add(new Choice(dev->T("Stats")))->OnClick.Handle(this, &JitCompareScreen::OnShowStats);
|
2015-07-01 21:26:55 +00:00
|
|
|
leftColumn->Add(new Choice(di->T("Back")))->OnClick.Handle<UIScreen>(this, &UIScreen::OnBack);
|
2015-06-30 03:28:31 +00:00
|
|
|
blockName_ = leftColumn->Add(new TextView(dev->T("No block")));
|
2014-11-16 15:45:24 +00:00
|
|
|
blockAddr_ = leftColumn->Add(new TextEdit("", "", new LayoutParams(FILL_PARENT, WRAP_CONTENT)));
|
|
|
|
blockAddr_->OnTextChange.Handle(this, &JitCompareScreen::OnAddressChange);
|
|
|
|
blockStats_ = leftColumn->Add(new TextView(""));
|
2014-10-12 16:44:16 +00:00
|
|
|
|
2017-03-22 01:34:52 +00:00
|
|
|
EventParams ignore{};
|
2014-10-12 16:44:16 +00:00
|
|
|
OnCurrentBlock(ignore);
|
2013-09-29 11:39:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void JitCompareScreen::UpdateDisasm() {
|
|
|
|
leftDisasm_->Clear();
|
|
|
|
rightDisasm_->Clear();
|
|
|
|
|
|
|
|
using namespace UI;
|
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-10 22:01:09 +00:00
|
|
|
|
2014-11-16 15:45:24 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
|
|
|
|
|
|
|
char temp[256];
|
|
|
|
snprintf(temp, sizeof(temp), "%i/%i", currentBlock_, blockCache->GetNumBlocks());
|
|
|
|
blockName_->SetText(temp);
|
|
|
|
|
|
|
|
if (currentBlock_ < 0 || currentBlock_ >= blockCache->GetNumBlocks()) {
|
2015-06-30 03:28:31 +00:00
|
|
|
leftDisasm_->Add(new TextView(dev->T("No block")));
|
|
|
|
rightDisasm_->Add(new TextView(dev->T("No block")));
|
2014-11-16 15:45:24 +00:00
|
|
|
blockStats_->SetText("");
|
2013-09-29 11:39:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
JitBlock *block = blockCache->GetBlock(currentBlock_);
|
|
|
|
|
2014-11-16 15:45:24 +00:00
|
|
|
snprintf(temp, sizeof(temp), "%08x", block->originalAddress);
|
|
|
|
blockAddr_->SetText(temp);
|
2013-09-30 08:10:34 +00:00
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
// Alright. First generate the MIPS disassembly.
|
|
|
|
|
2013-11-11 03:38:42 +00:00
|
|
|
// TODO: Need a way to communicate branch continuing.
|
2013-09-29 11:39:25 +00:00
|
|
|
for (u32 addr = block->originalAddress; addr <= block->originalAddress + block->originalSize * 4; addr += 4) {
|
|
|
|
char temp[256];
|
|
|
|
MIPSDisAsm(Memory::Read_Instruction(addr), addr, temp, true);
|
|
|
|
std::string mipsDis = temp;
|
2015-04-03 09:50:03 +00:00
|
|
|
leftDisasm_->Add(new TextView(mipsDis))->SetFocusable(true);
|
2013-09-29 11:39:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(ARM)
|
|
|
|
std::vector<std::string> targetDis = DisassembleArm2(block->normalEntry, block->codeSize);
|
2015-03-06 20:42:19 +00:00
|
|
|
#elif defined(ARM64)
|
|
|
|
std::vector<std::string> targetDis = DisassembleArm64(block->normalEntry, block->codeSize);
|
2015-05-25 12:26:40 +00:00
|
|
|
#elif defined(_M_IX86) || defined(_M_X64)
|
2014-11-09 11:03:04 +00:00
|
|
|
std::vector<std::string> targetDis = DisassembleX86(block->normalEntry, block->codeSize);
|
2014-10-11 16:06:52 +00:00
|
|
|
#endif
|
2015-05-25 12:26:40 +00:00
|
|
|
#if defined(ARM) || defined(ARM64) || defined(_M_IX86) || defined(_M_X64)
|
2014-11-09 11:03:04 +00:00
|
|
|
for (size_t i = 0; i < targetDis.size(); i++) {
|
2015-04-03 09:50:03 +00:00
|
|
|
rightDisasm_->Add(new TextView(targetDis[i]))->SetFocusable(true);
|
2014-10-11 16:06:52 +00:00
|
|
|
}
|
2015-05-25 12:26:40 +00:00
|
|
|
#endif
|
2014-11-16 15:45:24 +00:00
|
|
|
|
|
|
|
int numMips = leftDisasm_->GetNumSubviews();
|
|
|
|
int numHost = rightDisasm_->GetNumSubviews();
|
|
|
|
|
|
|
|
snprintf(temp, sizeof(temp), "%d to %d : %d%%", numMips, numHost, 100 * numHost / numMips);
|
|
|
|
blockStats_->SetText(temp);
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn JitCompareScreen::OnAddressChange(UI::EventParams &e) {
|
2015-03-07 16:58:15 +00:00
|
|
|
if (!MIPSComp::jit) {
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
2014-11-16 15:45:24 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
2016-05-08 19:56:52 +00:00
|
|
|
if (!blockCache)
|
|
|
|
return UI::EVENT_DONE;
|
2014-11-16 15:45:24 +00:00
|
|
|
u32 addr;
|
|
|
|
if (blockAddr_->GetText().size() > 8)
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
if (1 == sscanf(blockAddr_->GetText().c_str(), "%08x", &addr)) {
|
|
|
|
if (Memory::IsValidAddress(addr)) {
|
|
|
|
currentBlock_ = blockCache->GetBlockNumberFromStartAddress(addr);
|
|
|
|
UpdateDisasm();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return UI::EVENT_DONE;
|
2013-09-29 11:39:25 +00:00
|
|
|
}
|
|
|
|
|
2014-11-24 20:57:27 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnShowStats(UI::EventParams &e) {
|
2017-01-25 18:11:33 +00:00
|
|
|
if (!MIPSComp::jit) {
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2014-11-24 20:57:27 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
|
|
|
BlockCacheStats bcStats;
|
|
|
|
blockCache->ComputeStats(bcStats);
|
|
|
|
NOTICE_LOG(JIT, "Num blocks: %i", bcStats.numBlocks);
|
|
|
|
NOTICE_LOG(JIT, "Average Bloat: %0.2f%%", 100 * bcStats.avgBloat);
|
|
|
|
NOTICE_LOG(JIT, "Min Bloat: %0.2f%% (%08x)", 100 * bcStats.minBloat, bcStats.minBloatBlock);
|
|
|
|
NOTICE_LOG(JIT, "Max Bloat: %0.2f%% (%08x)", 100 * bcStats.maxBloat, bcStats.maxBloatBlock);
|
|
|
|
|
|
|
|
int ctr = 0, sz = (int)bcStats.bloatMap.size();
|
|
|
|
for (auto iter : bcStats.bloatMap) {
|
|
|
|
if (ctr < 10 || ctr > sz - 10) {
|
|
|
|
NOTICE_LOG(JIT, "%08x: %f", iter.second, iter.first);
|
|
|
|
} else if (ctr == 10) {
|
|
|
|
NOTICE_LOG(JIT, "...");
|
|
|
|
}
|
|
|
|
ctr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-29 06:35:03 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnSelectBlock(UI::EventParams &e) {
|
2015-06-30 03:28:31 +00:00
|
|
|
I18NCategory *dev = GetI18NCategory("Developer");
|
2014-12-10 22:01:09 +00:00
|
|
|
|
2015-06-30 03:28:31 +00:00
|
|
|
auto addressPrompt = new AddressPromptScreen(dev->T("Block address"));
|
2013-11-29 06:35:03 +00:00
|
|
|
addressPrompt->OnChoice.Handle(this, &JitCompareScreen::OnBlockAddress);
|
|
|
|
screenManager()->push(addressPrompt);
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2014-11-16 15:45:24 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnPrevBlock(UI::EventParams &e) {
|
|
|
|
currentBlock_--;
|
|
|
|
UpdateDisasm();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn JitCompareScreen::OnNextBlock(UI::EventParams &e) {
|
|
|
|
currentBlock_++;
|
|
|
|
UpdateDisasm();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-11-29 06:35:03 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnBlockAddress(UI::EventParams &e) {
|
2015-03-07 16:58:15 +00:00
|
|
|
if (!MIPSComp::jit) {
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-11-29 06:35:03 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
2016-05-08 19:56:52 +00:00
|
|
|
if (!blockCache)
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
|
2013-11-29 06:35:03 +00:00
|
|
|
if (Memory::IsValidAddress(e.a)) {
|
|
|
|
currentBlock_ = blockCache->GetBlockNumberFromStartAddress(e.a);
|
|
|
|
} else {
|
|
|
|
currentBlock_ = -1;
|
|
|
|
}
|
|
|
|
UpdateDisasm();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnRandomBlock(UI::EventParams &e) {
|
2015-03-07 16:58:15 +00:00
|
|
|
if (!MIPSComp::jit) {
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
2016-05-08 19:56:52 +00:00
|
|
|
if (!blockCache)
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
int numBlocks = blockCache->GetNumBlocks();
|
|
|
|
if (numBlocks > 0) {
|
|
|
|
currentBlock_ = rand() % numBlocks;
|
|
|
|
}
|
|
|
|
UpdateDisasm();
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
2013-11-07 12:37:19 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnRandomVFPUBlock(UI::EventParams &e) {
|
2015-03-16 23:54:56 +00:00
|
|
|
OnRandomBlock(IS_VFPU);
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UI::EventReturn JitCompareScreen::OnRandomFPUBlock(UI::EventParams &e) {
|
|
|
|
OnRandomBlock(IS_FPU);
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JitCompareScreen::OnRandomBlock(int flag) {
|
2015-03-07 16:58:15 +00:00
|
|
|
if (!MIPSComp::jit) {
|
2015-03-16 23:54:56 +00:00
|
|
|
return;
|
2015-03-07 16:58:15 +00:00
|
|
|
}
|
2013-11-07 12:37:19 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
2016-05-08 19:56:52 +00:00
|
|
|
if (!blockCache)
|
|
|
|
return;
|
|
|
|
|
2013-11-07 12:37:19 +00:00
|
|
|
int numBlocks = blockCache->GetNumBlocks();
|
|
|
|
if (numBlocks > 0) {
|
2015-03-16 23:54:56 +00:00
|
|
|
bool anyWanted = false;
|
2013-11-07 12:37:19 +00:00
|
|
|
int tries = 0;
|
2015-03-16 23:54:56 +00:00
|
|
|
while (!anyWanted && tries < 10000) {
|
2013-11-07 12:37:19 +00:00
|
|
|
currentBlock_ = rand() % numBlocks;
|
|
|
|
const JitBlock *b = blockCache->GetBlock(currentBlock_);
|
|
|
|
for (u32 addr = b->originalAddress; addr <= b->originalAddress + b->originalSize; addr += 4) {
|
|
|
|
MIPSOpcode opcode = Memory::Read_Instruction(addr);
|
2015-03-16 23:54:56 +00:00
|
|
|
if (MIPSGetInfo(opcode) & flag) {
|
2013-11-07 12:37:19 +00:00
|
|
|
char temp[256];
|
|
|
|
MIPSDisAsm(opcode, addr, temp);
|
2014-12-07 11:07:54 +00:00
|
|
|
// INFO_LOG(HLE, "Stopping VFPU instruction: %s", temp);
|
2015-03-16 23:54:56 +00:00
|
|
|
anyWanted = true;
|
2013-11-07 12:37:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tries++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
UpdateDisasm();
|
|
|
|
}
|
|
|
|
|
2013-09-29 11:39:25 +00:00
|
|
|
UI::EventReturn JitCompareScreen::OnCurrentBlock(UI::EventParams &e) {
|
2015-03-07 16:58:15 +00:00
|
|
|
if (!MIPSComp::jit) {
|
|
|
|
return UI::EVENT_DONE;
|
|
|
|
}
|
2013-09-29 11:39:25 +00:00
|
|
|
JitBlockCache *blockCache = MIPSComp::jit->GetBlockCache();
|
2016-05-08 19:56:52 +00:00
|
|
|
if (!blockCache)
|
|
|
|
return UI::EVENT_DONE;
|
2013-09-29 11:39:25 +00:00
|
|
|
std::vector<int> blockNum;
|
|
|
|
blockCache->GetBlockNumbersFromAddress(currentMIPS->pc, &blockNum);
|
|
|
|
if (blockNum.size() > 0) {
|
|
|
|
currentBlock_ = blockNum[0];
|
|
|
|
} else {
|
|
|
|
currentBlock_ = -1;
|
|
|
|
}
|
|
|
|
UpdateDisasm();
|
|
|
|
return UI::EVENT_DONE;
|
2013-09-29 17:59:55 +00:00
|
|
|
}
|
2015-05-13 20:28:02 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
void ShaderListScreen::ListShaders(DebugShaderType shaderType, UI::LinearLayout *view) {
|
|
|
|
using namespace UI;
|
|
|
|
std::vector<std::string> shaderIds_ = gpu->DebugGetShaderIDs(shaderType);
|
|
|
|
for (auto id : shaderIds_) {
|
|
|
|
Choice *choice = view->Add(new Choice(gpu->DebugGetShaderString(id, shaderType, SHADER_STRING_SHORT_DESC)));
|
|
|
|
choice->SetTag(id);
|
|
|
|
choice->OnClick.Handle(this, &ShaderListScreen::OnShaderClick);
|
2015-05-16 20:43:32 +00:00
|
|
|
}
|
2015-10-14 15:45:21 +00:00
|
|
|
}
|
2015-05-16 20:43:32 +00:00
|
|
|
|
2015-10-21 21:06:32 +00:00
|
|
|
struct { DebugShaderType type; const char *name; } shaderTypes[] = {
|
|
|
|
{ SHADER_TYPE_VERTEX, "Vertex" },
|
|
|
|
{ SHADER_TYPE_FRAGMENT, "Fragment" },
|
|
|
|
// { SHADER_TYPE_GEOMETRY, "Geometry" },
|
|
|
|
{ SHADER_TYPE_VERTEXLOADER, "VertexLoader" },
|
2017-04-04 09:09:29 +00:00
|
|
|
{ SHADER_TYPE_PIPELINE, "Pipeline" },
|
|
|
|
{ SHADER_TYPE_DEPAL, "Depal" },
|
2015-10-21 21:06:32 +00:00
|
|
|
};
|
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
void ShaderListScreen::CreateViews() {
|
|
|
|
using namespace UI;
|
2015-05-13 20:28:02 +00:00
|
|
|
|
2015-10-14 20:05:05 +00:00
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
LinearLayout *layout = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
root_ = layout;
|
2015-05-13 20:28:02 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
tabs_ = new TabHolder(ORIENT_HORIZONTAL, 40, new LinearLayoutParams(1.0));
|
2016-01-23 06:38:21 +00:00
|
|
|
tabs_->SetTag("DevShaderList");
|
2015-10-14 15:45:21 +00:00
|
|
|
layout->Add(tabs_);
|
2015-10-14 20:05:05 +00:00
|
|
|
layout->Add(new Button(di->T("Back")))->OnClick.Handle<UIScreen>(this, &UIScreen::OnBack);
|
2015-11-18 13:18:35 +00:00
|
|
|
for (size_t i = 0; i < ARRAY_SIZE(shaderTypes); i++) {
|
2015-10-21 21:06:32 +00:00
|
|
|
ScrollView *scroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(1.0));
|
|
|
|
LinearLayout *shaderList = new LinearLayout(ORIENT_VERTICAL, new LayoutParams(FILL_PARENT, WRAP_CONTENT));
|
|
|
|
ListShaders(shaderTypes[i].type, shaderList);
|
|
|
|
scroll->Add(shaderList);
|
|
|
|
tabs_->AddTab(shaderTypes[i].name, scroll);
|
|
|
|
}
|
2015-10-14 15:45:21 +00:00
|
|
|
}
|
2015-05-13 20:28:02 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
UI::EventReturn ShaderListScreen::OnShaderClick(UI::EventParams &e) {
|
|
|
|
using namespace UI;
|
|
|
|
std::string id = e.v->Tag();
|
2015-10-21 21:06:32 +00:00
|
|
|
DebugShaderType type = shaderTypes[tabs_->GetCurrentTab()].type;
|
2015-10-14 15:45:21 +00:00
|
|
|
screenManager()->push(new ShaderViewScreen(id, type));
|
|
|
|
return EVENT_DONE;
|
|
|
|
}
|
2015-05-13 21:07:19 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
void ShaderViewScreen::CreateViews() {
|
|
|
|
using namespace UI;
|
2015-05-13 21:07:19 +00:00
|
|
|
|
2015-10-14 20:05:05 +00:00
|
|
|
I18NCategory *di = GetI18NCategory("Dialog");
|
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
LinearLayout *layout = new LinearLayout(ORIENT_VERTICAL);
|
|
|
|
root_ = layout;
|
2015-05-13 20:28:02 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
layout->Add(new TextView(gpu->DebugGetShaderString(id_, type_, SHADER_STRING_SHORT_DESC)));
|
2015-05-13 21:07:19 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
ScrollView *scroll = new ScrollView(ORIENT_VERTICAL, new LinearLayoutParams(1.0));
|
2016-01-23 06:52:13 +00:00
|
|
|
scroll->SetTag("DevShaderView");
|
2015-10-14 15:45:21 +00:00
|
|
|
layout->Add(scroll);
|
2015-07-03 19:43:02 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
LinearLayout *lineLayout = new LinearLayout(ORIENT_VERTICAL, new LinearLayoutParams(FILL_PARENT, WRAP_CONTENT));
|
|
|
|
lineLayout->SetSpacing(0.0);
|
|
|
|
scroll->Add(lineLayout);
|
2015-05-13 21:07:19 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
std::vector<std::string> lines;
|
|
|
|
SplitString(gpu->DebugGetShaderString(id_, type_, SHADER_STRING_SOURCE_CODE), '\n', lines);
|
2015-05-25 22:55:25 +00:00
|
|
|
|
2015-10-14 15:45:21 +00:00
|
|
|
for (auto line : lines) {
|
|
|
|
lineLayout->Add(new TextView(line, FLAG_DYNAMIC_ASCII, true));
|
2015-05-25 22:55:25 +00:00
|
|
|
}
|
|
|
|
|
2015-10-14 20:05:05 +00:00
|
|
|
layout->Add(new Button(di->T("Back")))->OnClick.Handle<UIScreen>(this, &UIScreen::OnBack);
|
2015-05-13 20:28:02 +00:00
|
|
|
}
|