mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 15:31:17 +01:00
cfd25f1d7c
We can move the construction of the Diff instance into the body of the if statement, so that we only construct this if the condition is true. While we're at it, we can move the symbol description string into the instance, getting rid of a copy. The function itself can also be const qualified.
536 lines
19 KiB
C++
536 lines
19 KiB
C++
// Copyright 2022 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "DolphinQt/Debugger/CodeDiffDialog.h"
|
|
|
|
#include <algorithm>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include <QGuiApplication>
|
|
#include <QHBoxLayout>
|
|
#include <QLabel>
|
|
#include <QMenu>
|
|
#include <QPushButton>
|
|
#include <QStyleHints>
|
|
#include <QTableWidget>
|
|
#include <QVBoxLayout>
|
|
|
|
#include "Common/StringUtil.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/Debugger/PPCDebugInterface.h"
|
|
#include "Core/HW/CPU.h"
|
|
#include "Core/PowerPC/JitInterface.h"
|
|
#include "Core/PowerPC/MMU.h"
|
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
|
#include "Core/PowerPC/PowerPC.h"
|
|
#include "Core/PowerPC/Profiler.h"
|
|
#include "Core/System.h"
|
|
|
|
#include "DolphinQt/Debugger/CodeWidget.h"
|
|
#include "DolphinQt/Host.h"
|
|
#include "DolphinQt/QtUtils/ModalMessageBox.h"
|
|
#include "DolphinQt/Settings.h"
|
|
|
|
static const QString RECORD_BUTTON_STYLESHEET =
|
|
QStringLiteral("QPushButton:checked { background-color: rgb(150, 0, 0); border-style: solid; "
|
|
"border-width: 3px; border-color: rgb(150,0,0); color: rgb(255, 255, 255);}");
|
|
|
|
CodeDiffDialog::CodeDiffDialog(CodeWidget* parent) : QDialog(parent), m_code_widget(parent)
|
|
{
|
|
setWindowTitle(tr("Code Diff Tool"));
|
|
CreateWidgets();
|
|
auto& settings = Settings::GetQSettings();
|
|
restoreGeometry(settings.value(QStringLiteral("diffdialog/geometry")).toByteArray());
|
|
ConnectWidgets();
|
|
}
|
|
|
|
void CodeDiffDialog::reject()
|
|
{
|
|
ClearData();
|
|
auto& settings = Settings::GetQSettings();
|
|
settings.setValue(QStringLiteral("diffdialog/geometry"), saveGeometry());
|
|
QDialog::reject();
|
|
}
|
|
|
|
void CodeDiffDialog::CreateWidgets()
|
|
{
|
|
auto* btns_layout = new QGridLayout;
|
|
m_exclude_btn = new QPushButton(tr("Code did not get executed"));
|
|
m_include_btn = new QPushButton(tr("Code has been executed"));
|
|
m_record_btn = new QPushButton(tr("Start Recording"));
|
|
m_record_btn->setCheckable(true);
|
|
m_record_btn->setStyleSheet(RECORD_BUTTON_STYLESHEET);
|
|
|
|
m_exclude_btn->setEnabled(false);
|
|
m_include_btn->setEnabled(false);
|
|
|
|
btns_layout->addWidget(m_exclude_btn, 0, 0);
|
|
btns_layout->addWidget(m_include_btn, 0, 1);
|
|
btns_layout->addWidget(m_record_btn, 0, 2);
|
|
|
|
auto* labels_layout = new QHBoxLayout;
|
|
m_exclude_size_label = new QLabel(tr("Excluded: 0"));
|
|
m_include_size_label = new QLabel(tr("Included: 0"));
|
|
|
|
btns_layout->addWidget(m_exclude_size_label, 1, 0);
|
|
btns_layout->addWidget(m_include_size_label, 1, 1);
|
|
|
|
m_matching_results_table = new QTableWidget();
|
|
m_matching_results_table->setColumnCount(5);
|
|
m_matching_results_table->setHorizontalHeaderLabels(
|
|
{tr("Address"), tr("Total Hits"), tr("Hits"), tr("Symbol"), tr("Inspected")});
|
|
m_matching_results_table->setSelectionMode(QAbstractItemView::SingleSelection);
|
|
m_matching_results_table->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
m_matching_results_table->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
m_matching_results_table->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
m_matching_results_table->setColumnWidth(0, 60);
|
|
m_matching_results_table->setColumnWidth(1, 60);
|
|
m_matching_results_table->setColumnWidth(2, 4);
|
|
m_matching_results_table->setColumnWidth(3, 210);
|
|
m_matching_results_table->setColumnWidth(4, 65);
|
|
m_reset_btn = new QPushButton(tr("Reset All"));
|
|
m_reset_btn->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
m_help_btn = new QPushButton(tr("Help"));
|
|
m_help_btn->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
auto* help_reset_layout = new QHBoxLayout;
|
|
help_reset_layout->addWidget(m_reset_btn, 0, Qt::AlignLeft);
|
|
help_reset_layout->addWidget(m_help_btn, 0, Qt::AlignRight);
|
|
|
|
auto* layout = new QVBoxLayout();
|
|
layout->addLayout(btns_layout);
|
|
layout->addLayout(labels_layout);
|
|
layout->addWidget(m_matching_results_table);
|
|
layout->addLayout(help_reset_layout);
|
|
|
|
setLayout(layout);
|
|
resize(515, 400);
|
|
}
|
|
|
|
void CodeDiffDialog::ConnectWidgets()
|
|
{
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 5, 0)
|
|
connect(QGuiApplication::styleHints(), &QStyleHints::colorSchemeChanged, this,
|
|
[this](Qt::ColorScheme colorScheme) {
|
|
m_record_btn->setStyleSheet(RECORD_BUTTON_STYLESHEET);
|
|
});
|
|
#endif
|
|
|
|
connect(m_record_btn, &QPushButton::toggled, this, &CodeDiffDialog::OnRecord);
|
|
connect(m_include_btn, &QPushButton::pressed, [this]() { Update(true); });
|
|
connect(m_exclude_btn, &QPushButton::pressed, [this]() { Update(false); });
|
|
connect(m_matching_results_table, &QTableWidget::itemClicked, [this]() { OnClickItem(); });
|
|
connect(m_reset_btn, &QPushButton::pressed, this, &CodeDiffDialog::ClearData);
|
|
connect(m_help_btn, &QPushButton::pressed, this, &CodeDiffDialog::InfoDisp);
|
|
connect(m_matching_results_table, &CodeDiffDialog::customContextMenuRequested, this,
|
|
&CodeDiffDialog::OnContextMenu);
|
|
}
|
|
|
|
void CodeDiffDialog::OnClickItem()
|
|
{
|
|
UpdateItem();
|
|
auto address = m_matching_results_table->currentItem()->data(Qt::UserRole).toUInt();
|
|
m_code_widget->SetAddress(address, CodeViewWidget::SetAddressUpdate::WithDetailedUpdate);
|
|
}
|
|
|
|
void CodeDiffDialog::ClearData()
|
|
{
|
|
if (m_record_btn->isChecked())
|
|
m_record_btn->toggle();
|
|
ClearBlockCache();
|
|
m_matching_results_table->clear();
|
|
m_matching_results_table->setRowCount(0);
|
|
m_matching_results_table->setHorizontalHeaderLabels(
|
|
{tr("Address"), tr("Total Hits"), tr("Hits"), tr("Symbol"), tr("Inspected")});
|
|
m_matching_results_table->setEditTriggers(QAbstractItemView::EditTrigger::NoEditTriggers);
|
|
m_exclude_size_label->setText(tr("Excluded: %1").arg(0));
|
|
m_include_size_label->setText(tr("Included: %1").arg(0));
|
|
m_exclude_btn->setEnabled(false);
|
|
m_include_btn->setEnabled(false);
|
|
m_include_active = false;
|
|
// Swap is used instead of clear for efficiency in the case of huge m_include/m_exclude
|
|
std::vector<Diff>().swap(m_include);
|
|
std::vector<Diff>().swap(m_exclude);
|
|
Core::System::GetInstance().GetJitInterface().SetProfilingState(
|
|
JitInterface::ProfilingState::Disabled);
|
|
}
|
|
|
|
void CodeDiffDialog::ClearBlockCache()
|
|
{
|
|
Core::State old_state = Core::GetState();
|
|
|
|
if (old_state == Core::State::Running)
|
|
Core::SetState(Core::State::Paused);
|
|
|
|
Core::System::GetInstance().GetJitInterface().ClearCache();
|
|
|
|
if (old_state == Core::State::Running)
|
|
Core::SetState(Core::State::Running);
|
|
}
|
|
|
|
void CodeDiffDialog::OnRecord(bool enabled)
|
|
{
|
|
if (m_failed_requirements)
|
|
{
|
|
m_failed_requirements = false;
|
|
return;
|
|
}
|
|
|
|
if (Core::GetState() == Core::State::Uninitialized)
|
|
{
|
|
ModalMessageBox::information(this, tr("Code Diff Tool"),
|
|
tr("Emulation must be started to record."));
|
|
m_failed_requirements = true;
|
|
m_record_btn->setChecked(false);
|
|
return;
|
|
}
|
|
|
|
if (g_symbolDB.IsEmpty())
|
|
{
|
|
ModalMessageBox::warning(
|
|
this, tr("Code Diff Tool"),
|
|
tr("Symbol map not found.\n\nIf one does not exist, you can generate one from "
|
|
"the Menu bar:\nSymbols -> Generate Symbols From ->\n\tAddress | Signature "
|
|
"Database | RSO Modules"));
|
|
m_failed_requirements = true;
|
|
m_record_btn->setChecked(false);
|
|
return;
|
|
}
|
|
|
|
JitInterface::ProfilingState state;
|
|
|
|
if (enabled)
|
|
{
|
|
ClearBlockCache();
|
|
m_record_btn->setText(tr("Stop Recording"));
|
|
state = JitInterface::ProfilingState::Enabled;
|
|
m_exclude_btn->setEnabled(true);
|
|
m_include_btn->setEnabled(true);
|
|
}
|
|
else
|
|
{
|
|
ClearBlockCache();
|
|
m_record_btn->setText(tr("Start Recording"));
|
|
state = JitInterface::ProfilingState::Disabled;
|
|
m_exclude_btn->setEnabled(false);
|
|
m_include_btn->setEnabled(false);
|
|
}
|
|
|
|
m_record_btn->update();
|
|
Core::System::GetInstance().GetJitInterface().SetProfilingState(state);
|
|
}
|
|
|
|
void CodeDiffDialog::OnInclude()
|
|
{
|
|
const auto recorded_symbols = CalculateSymbolsFromProfile();
|
|
|
|
if (recorded_symbols.empty())
|
|
return;
|
|
|
|
if (m_include.empty() && m_exclude.empty())
|
|
{
|
|
m_include = recorded_symbols;
|
|
m_include_active = true;
|
|
}
|
|
else if (m_include.empty())
|
|
{
|
|
// If include becomes empty after having items on it, don't refill it until after a reset.
|
|
if (m_include_active)
|
|
return;
|
|
|
|
// If we are building include for the first time and we have an exlcude list, then include =
|
|
// recorded - excluded.
|
|
m_include = recorded_symbols;
|
|
RemoveMatchingSymbolsFromIncludes(m_exclude);
|
|
m_include_active = true;
|
|
}
|
|
else
|
|
{
|
|
// If include already exists, keep items that are in both include and recorded. Exclude list
|
|
// becomes irrelevant.
|
|
RemoveMissingSymbolsFromIncludes(recorded_symbols);
|
|
}
|
|
}
|
|
|
|
void CodeDiffDialog::OnExclude()
|
|
{
|
|
const auto recorded_symbols = CalculateSymbolsFromProfile();
|
|
if (m_include.empty() && m_exclude.empty())
|
|
{
|
|
m_exclude = recorded_symbols;
|
|
}
|
|
else if (m_include.empty())
|
|
{
|
|
// If there is only an exclude list, update it.
|
|
for (auto& iter : recorded_symbols)
|
|
{
|
|
auto pos = std::lower_bound(m_exclude.begin(), m_exclude.end(), iter.symbol);
|
|
|
|
if (pos == m_exclude.end() || pos->symbol != iter.symbol)
|
|
m_exclude.insert(pos, iter);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If include already exists, the exclude list will have been used to trim it, so the exclude
|
|
// list is now irrelevant, as anythng not on the include list is effectively excluded.
|
|
// Exclude/subtract recorded items from the include list.
|
|
RemoveMatchingSymbolsFromIncludes(recorded_symbols);
|
|
}
|
|
}
|
|
|
|
std::vector<Diff> CodeDiffDialog::CalculateSymbolsFromProfile() const
|
|
{
|
|
Profiler::ProfileStats prof_stats;
|
|
auto& blockstats = prof_stats.block_stats;
|
|
Core::System::GetInstance().GetJitInterface().GetProfileResults(&prof_stats);
|
|
std::vector<Diff> current;
|
|
current.reserve(20000);
|
|
|
|
// Convert blockstats to smaller struct Diff. Exclude repeat functions via symbols.
|
|
for (const auto& iter : blockstats)
|
|
{
|
|
std::string symbol = g_symbolDB.GetDescription(iter.addr);
|
|
if (!std::any_of(current.begin(), current.end(),
|
|
[&symbol](const Diff& v) { return v.symbol == symbol; }))
|
|
{
|
|
current.push_back(Diff{
|
|
.addr = iter.addr,
|
|
.symbol = std::move(symbol),
|
|
.hits = static_cast<u32>(iter.run_count),
|
|
.total_hits = static_cast<u32>(iter.run_count),
|
|
});
|
|
}
|
|
}
|
|
|
|
std::sort(current.begin(), current.end(),
|
|
[](const Diff& v1, const Diff& v2) { return (v1.symbol < v2.symbol); });
|
|
|
|
return current;
|
|
}
|
|
|
|
void CodeDiffDialog::RemoveMissingSymbolsFromIncludes(const std::vector<Diff>& symbol_diff)
|
|
{
|
|
m_include.erase(std::remove_if(m_include.begin(), m_include.end(),
|
|
[&](const Diff& v) {
|
|
auto arg = std::none_of(
|
|
symbol_diff.begin(), symbol_diff.end(), [&](const Diff& p) {
|
|
return p.symbol == v.symbol || p.addr == v.addr;
|
|
});
|
|
return arg;
|
|
}),
|
|
m_include.end());
|
|
for (auto& original_includes : m_include)
|
|
{
|
|
auto pos = std::lower_bound(symbol_diff.begin(), symbol_diff.end(), original_includes.symbol);
|
|
if (pos != symbol_diff.end() &&
|
|
(pos->symbol == original_includes.symbol || pos->addr == original_includes.addr))
|
|
{
|
|
original_includes.total_hits += pos->hits;
|
|
original_includes.hits = pos->hits;
|
|
}
|
|
}
|
|
}
|
|
|
|
void CodeDiffDialog::RemoveMatchingSymbolsFromIncludes(const std::vector<Diff>& symbol_list)
|
|
{
|
|
m_include.erase(std::remove_if(m_include.begin(), m_include.end(),
|
|
[&](const Diff& i) {
|
|
return std::any_of(
|
|
symbol_list.begin(), symbol_list.end(), [&](const Diff& s) {
|
|
return i.symbol == s.symbol || i.addr == s.addr;
|
|
});
|
|
}),
|
|
m_include.end());
|
|
}
|
|
|
|
void CodeDiffDialog::Update(bool include)
|
|
{
|
|
// Wrap everything in a pause
|
|
Core::State old_state = Core::GetState();
|
|
if (old_state == Core::State::Running)
|
|
Core::SetState(Core::State::Paused);
|
|
|
|
// Main process
|
|
if (include)
|
|
{
|
|
OnInclude();
|
|
}
|
|
else
|
|
{
|
|
OnExclude();
|
|
}
|
|
|
|
const auto create_item = [](const QString& string = {}, const u32 address = 0x00000000) {
|
|
QTableWidgetItem* item = new QTableWidgetItem(string);
|
|
item->setData(Qt::UserRole, address);
|
|
item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable);
|
|
return item;
|
|
};
|
|
|
|
int i = 0;
|
|
m_matching_results_table->clear();
|
|
m_matching_results_table->setRowCount(i);
|
|
m_matching_results_table->setHorizontalHeaderLabels(
|
|
{tr("Address"), tr("Total Hits"), tr("Hits"), tr("Symbol"), tr("Inspected")});
|
|
|
|
for (auto& iter : m_include)
|
|
{
|
|
m_matching_results_table->setRowCount(i + 1);
|
|
|
|
QString fix_sym = QString::fromStdString(iter.symbol);
|
|
fix_sym.replace(QStringLiteral("\t"), QStringLiteral(" "));
|
|
|
|
m_matching_results_table->setItem(
|
|
i, 0, create_item(QStringLiteral("%1").arg(iter.addr, 1, 16), iter.addr));
|
|
m_matching_results_table->setItem(
|
|
i, 1, create_item(QStringLiteral("%1").arg(iter.total_hits), iter.addr));
|
|
m_matching_results_table->setItem(i, 2,
|
|
create_item(QStringLiteral("%1").arg(iter.hits), iter.addr));
|
|
m_matching_results_table->setItem(i, 3,
|
|
create_item(QStringLiteral("%1").arg(fix_sym), iter.addr));
|
|
m_matching_results_table->setItem(i, 4, create_item(QStringLiteral(""), iter.addr));
|
|
i++;
|
|
}
|
|
|
|
// If we have ruled out all functions from being included.
|
|
if (m_include_active && m_include.empty())
|
|
{
|
|
m_matching_results_table->setRowCount(1);
|
|
m_matching_results_table->setItem(0, 3, create_item(tr("No possible functions left. Reset.")));
|
|
}
|
|
|
|
m_exclude_size_label->setText(tr("Excluded: %1").arg(m_exclude.size()));
|
|
m_include_size_label->setText(tr("Included: %1").arg(m_include.size()));
|
|
|
|
Core::System::GetInstance().GetJitInterface().ClearCache();
|
|
if (old_state == Core::State::Running)
|
|
Core::SetState(Core::State::Running);
|
|
}
|
|
|
|
void CodeDiffDialog::InfoDisp()
|
|
{
|
|
ModalMessageBox::information(
|
|
this, tr("Code Diff Tool Help"),
|
|
tr("Used to find functions based on when they should be running.\nSimilar to Cheat Engine "
|
|
"Ultimap.\n"
|
|
"A symbol map must be loaded prior to use.\n"
|
|
"Include/Exclude lists will persist on ending/restarting emulation.\nThese lists "
|
|
"will not persist on Dolphin close."
|
|
"\n\n'Start Recording': "
|
|
"keeps track of what functions run.\n'Stop Recording': erases current "
|
|
"recording without any change to the lists.\n'Code did not get executed': click while "
|
|
"recording, will add recorded functions to an exclude "
|
|
"list, then reset the recording list.\n'Code has been executed': click while recording, "
|
|
"will add recorded function to an include list, then reset the recording list.\n\nAfter "
|
|
"you use "
|
|
"both exclude and include once, the exclude list will be subtracted from the include "
|
|
"list "
|
|
"and any includes left over will be displayed.\nYou can continue to use "
|
|
"'Code did not get executed'/'Code has been executed' to narrow down the "
|
|
"results."));
|
|
ModalMessageBox::information(
|
|
this, tr("Code Diff Tool Help"),
|
|
tr("Example:\n"
|
|
"You want to find a function that runs when HP is modified.\n1. Start recording and "
|
|
"play the game without letting HP be modified, then press 'Code did not get "
|
|
"executed'.\n2. Immediately gain/lose HP and press 'Code has been executed'.\n3. Repeat "
|
|
"1 or 2 to "
|
|
"narrow down the results.\nIncludes (Code has been executed) should "
|
|
"have short recordings focusing on what you want.\n\nPressing 'Code has been "
|
|
"executed' twice will only keep functions that ran for both recordings. Hits will update "
|
|
"to reflect the last recording's "
|
|
"number of Hits. Total Hits will reflect the total number of "
|
|
"times a function has been executed until the lists are cleared with Reset.\n\nRight "
|
|
"click -> 'Set blr' will place a "
|
|
"blr at the top of the symbol.\n"));
|
|
}
|
|
|
|
void CodeDiffDialog::OnContextMenu()
|
|
{
|
|
if (m_matching_results_table->currentItem() == nullptr)
|
|
return;
|
|
UpdateItem();
|
|
QMenu* menu = new QMenu(this);
|
|
menu->addAction(tr("&Go to start of function"), this, &CodeDiffDialog::OnGoTop);
|
|
menu->addAction(tr("Set &blr"), this, &CodeDiffDialog::OnSetBLR);
|
|
menu->addAction(tr("&Delete"), this, &CodeDiffDialog::OnDelete);
|
|
menu->exec(QCursor::pos());
|
|
}
|
|
|
|
void CodeDiffDialog::OnGoTop()
|
|
{
|
|
auto item = m_matching_results_table->currentItem();
|
|
if (!item)
|
|
return;
|
|
Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(item->data(Qt::UserRole).toUInt());
|
|
if (!symbol)
|
|
return;
|
|
m_code_widget->SetAddress(symbol->address, CodeViewWidget::SetAddressUpdate::WithDetailedUpdate);
|
|
}
|
|
|
|
void CodeDiffDialog::OnDelete()
|
|
{
|
|
// Delete from include list and qtable widget
|
|
auto item = m_matching_results_table->currentItem();
|
|
if (!item)
|
|
return;
|
|
int row = m_matching_results_table->row(item);
|
|
if (row == -1)
|
|
return;
|
|
// TODO: If/when sorting is ever added, .erase needs to find item position instead; leaving as is
|
|
// for performance
|
|
if (!m_include.empty())
|
|
{
|
|
m_include.erase(m_include.begin() + row);
|
|
}
|
|
m_matching_results_table->removeRow(row);
|
|
}
|
|
|
|
void CodeDiffDialog::OnSetBLR()
|
|
{
|
|
auto item = m_matching_results_table->currentItem();
|
|
if (!item)
|
|
return;
|
|
|
|
Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(item->data(Qt::UserRole).toUInt());
|
|
if (!symbol)
|
|
return;
|
|
|
|
{
|
|
auto& system = Core::System::GetInstance();
|
|
Core::CPUThreadGuard guard(system);
|
|
system.GetPowerPC().GetDebugInterface().SetPatch(guard, symbol->address, 0x4E800020);
|
|
}
|
|
|
|
int row = item->row();
|
|
m_matching_results_table->item(row, 0)->setForeground(QBrush(Qt::red));
|
|
m_matching_results_table->item(row, 1)->setForeground(QBrush(Qt::red));
|
|
m_matching_results_table->item(row, 2)->setForeground(QBrush(Qt::red));
|
|
m_matching_results_table->item(row, 3)->setForeground(QBrush(Qt::red));
|
|
m_matching_results_table->item(row, 4)->setForeground(QBrush(Qt::red));
|
|
m_matching_results_table->item(row, 4)->setText(QStringLiteral("X"));
|
|
|
|
m_code_widget->Update();
|
|
}
|
|
|
|
void CodeDiffDialog::UpdateItem()
|
|
{
|
|
QTableWidgetItem* item = m_matching_results_table->currentItem();
|
|
if (!item)
|
|
return;
|
|
|
|
int row = m_matching_results_table->row(item);
|
|
if (row == -1)
|
|
return;
|
|
uint address = item->data(Qt::UserRole).toUInt();
|
|
|
|
auto symbolName = g_symbolDB.GetDescription(address);
|
|
if (symbolName == " --- ")
|
|
return;
|
|
|
|
QString newName =
|
|
QString::fromStdString(symbolName).replace(QStringLiteral("\t"), QStringLiteral(" "));
|
|
m_matching_results_table->item(row, 3)->setText(newName);
|
|
}
|