2018-10-16 21:47:02 +00:00
|
|
|
/*
|
|
|
|
This file is part of solidity.
|
|
|
|
|
|
|
|
solidity 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, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
solidity 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 for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2020-07-17 14:54:12 +00:00
|
|
|
// SPDX-License-Identifier: GPL-3.0
|
2018-10-16 21:47:02 +00:00
|
|
|
/**
|
|
|
|
* Optimiser component that turns subsequent assignments to variable declarations
|
|
|
|
* and assignments.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <libyul/optimiser/SSATransform.h>
|
|
|
|
|
|
|
|
#include <libyul/optimiser/NameCollector.h>
|
|
|
|
#include <libyul/optimiser/NameDispenser.h>
|
2020-10-29 14:00:27 +00:00
|
|
|
#include <libyul/AST.h>
|
2018-10-16 21:47:02 +00:00
|
|
|
|
2020-01-06 10:52:23 +00:00
|
|
|
#include <libsolutil/CommonData.h>
|
2018-10-16 21:47:02 +00:00
|
|
|
|
2020-02-19 16:34:08 +00:00
|
|
|
#include <libyul/optimiser/TypeInfo.h>
|
|
|
|
|
2018-10-16 21:47:02 +00:00
|
|
|
using namespace std;
|
2019-12-11 16:31:36 +00:00
|
|
|
using namespace solidity;
|
|
|
|
using namespace solidity::yul;
|
|
|
|
using namespace solidity::langutil;
|
2018-10-16 21:47:02 +00:00
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
namespace
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
/**
|
|
|
|
* First step of SSA transform: Introduces new SSA variables for each assignment or
|
|
|
|
* declaration of a variable to be replaced.
|
|
|
|
*/
|
|
|
|
class IntroduceSSA: public ASTModifier
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2019-09-12 13:19:11 +00:00
|
|
|
public:
|
2020-02-19 16:34:08 +00:00
|
|
|
explicit IntroduceSSA(
|
|
|
|
NameDispenser& _nameDispenser,
|
|
|
|
set<YulString> const& _variablesToReplace,
|
|
|
|
TypeInfo& _typeInfo
|
|
|
|
):
|
|
|
|
m_nameDispenser(_nameDispenser),
|
|
|
|
m_variablesToReplace(_variablesToReplace),
|
|
|
|
m_typeInfo(_typeInfo)
|
2019-09-12 13:19:11 +00:00
|
|
|
{ }
|
2018-10-16 21:47:02 +00:00
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
void operator()(Block& _block) override;
|
2018-10-16 21:47:02 +00:00
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
private:
|
|
|
|
NameDispenser& m_nameDispenser;
|
|
|
|
set<YulString> const& m_variablesToReplace;
|
2020-02-19 16:34:08 +00:00
|
|
|
TypeInfo const& m_typeInfo;
|
2019-09-12 13:19:11 +00:00
|
|
|
};
|
2018-10-16 21:47:02 +00:00
|
|
|
|
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
void IntroduceSSA::operator()(Block& _block)
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2019-12-11 16:31:36 +00:00
|
|
|
util::iterateReplacing(
|
2018-10-16 21:47:02 +00:00
|
|
|
_block.statements,
|
2019-10-28 10:39:30 +00:00
|
|
|
[&](Statement& _s) -> std::optional<vector<Statement>>
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2019-11-19 15:42:49 +00:00
|
|
|
if (holds_alternative<VariableDeclaration>(_s))
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2019-11-19 15:42:49 +00:00
|
|
|
VariableDeclaration& varDecl = std::get<VariableDeclaration>(_s);
|
2018-10-16 21:47:02 +00:00
|
|
|
if (varDecl.value)
|
|
|
|
visit(*varDecl.value);
|
2019-04-03 18:16:30 +00:00
|
|
|
|
|
|
|
bool needToReplaceSome = false;
|
|
|
|
for (auto const& var: varDecl.variables)
|
|
|
|
if (m_variablesToReplace.count(var.name))
|
|
|
|
needToReplaceSome = true;
|
|
|
|
if (!needToReplaceSome)
|
2018-10-16 21:47:02 +00:00
|
|
|
return {};
|
2019-04-03 18:16:30 +00:00
|
|
|
|
2019-01-09 13:05:03 +00:00
|
|
|
// Replace "let a := v" by "let a_1 := v let a := a_1"
|
2019-04-03 18:16:30 +00:00
|
|
|
// Replace "let a, b := v" by "let a_1, b_1 := v let a := a_1 let b := b_2"
|
2021-04-27 14:53:04 +00:00
|
|
|
shared_ptr<DebugData const> debugData = varDecl.debugData;
|
2019-04-03 18:16:30 +00:00
|
|
|
vector<Statement> statements;
|
2021-04-27 14:53:04 +00:00
|
|
|
statements.emplace_back(VariableDeclaration{debugData, {}, std::move(varDecl.value)});
|
2019-04-03 18:16:30 +00:00
|
|
|
TypedNameList newVariables;
|
|
|
|
for (auto const& var: varDecl.variables)
|
|
|
|
{
|
|
|
|
YulString oldName = var.name;
|
2019-09-12 13:19:11 +00:00
|
|
|
YulString newName = m_nameDispenser.newName(oldName);
|
2021-04-27 14:53:04 +00:00
|
|
|
newVariables.emplace_back(TypedName{debugData, newName, var.type});
|
2019-04-03 18:16:30 +00:00
|
|
|
statements.emplace_back(VariableDeclaration{
|
2021-04-27 14:53:04 +00:00
|
|
|
debugData,
|
|
|
|
{TypedName{debugData, oldName, var.type}},
|
|
|
|
make_unique<Expression>(Identifier{debugData, newName})
|
2019-04-03 18:16:30 +00:00
|
|
|
});
|
|
|
|
}
|
2019-11-19 15:42:49 +00:00
|
|
|
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
2019-05-14 14:44:28 +00:00
|
|
|
return { std::move(statements) };
|
2018-10-16 21:47:02 +00:00
|
|
|
}
|
2019-11-19 15:42:49 +00:00
|
|
|
else if (holds_alternative<Assignment>(_s))
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2019-11-19 15:42:49 +00:00
|
|
|
Assignment& assignment = std::get<Assignment>(_s);
|
2018-10-16 21:47:02 +00:00
|
|
|
visit(*assignment.value);
|
2019-04-03 18:16:30 +00:00
|
|
|
for (auto const& var: assignment.variableNames)
|
|
|
|
assertThrow(m_variablesToReplace.count(var.name), OptimizerException, "");
|
|
|
|
|
2018-10-16 21:47:02 +00:00
|
|
|
// Replace "a := v" by "let a_1 := v a := v"
|
2019-04-03 18:16:30 +00:00
|
|
|
// Replace "a, b := v" by "let a_1, b_1 := v a := a_1 b := b_2"
|
2021-04-27 14:53:04 +00:00
|
|
|
std::shared_ptr<DebugData const> debugData = assignment.debugData;
|
2019-04-03 18:16:30 +00:00
|
|
|
vector<Statement> statements;
|
2021-04-27 14:53:04 +00:00
|
|
|
statements.emplace_back(VariableDeclaration{debugData, {}, std::move(assignment.value)});
|
2019-04-03 18:16:30 +00:00
|
|
|
TypedNameList newVariables;
|
|
|
|
for (auto const& var: assignment.variableNames)
|
|
|
|
{
|
|
|
|
YulString oldName = var.name;
|
2019-09-12 13:19:11 +00:00
|
|
|
YulString newName = m_nameDispenser.newName(oldName);
|
2021-04-27 14:53:04 +00:00
|
|
|
newVariables.emplace_back(TypedName{debugData,
|
2020-02-19 16:34:08 +00:00
|
|
|
newName,
|
|
|
|
m_typeInfo.typeOfVariable(oldName)
|
|
|
|
});
|
2019-04-03 18:16:30 +00:00
|
|
|
statements.emplace_back(Assignment{
|
2021-04-27 14:53:04 +00:00
|
|
|
debugData,
|
|
|
|
{Identifier{debugData, oldName}},
|
|
|
|
make_unique<Expression>(Identifier{debugData, newName})
|
2019-04-03 18:16:30 +00:00
|
|
|
});
|
|
|
|
}
|
2019-11-19 15:42:49 +00:00
|
|
|
std::get<VariableDeclaration>(statements.front()).variables = std::move(newVariables);
|
2019-05-14 14:44:28 +00:00
|
|
|
return { std::move(statements) };
|
2018-10-16 21:47:02 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
visit(_s);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
);
|
2019-09-12 13:19:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-07 19:14:25 +00:00
|
|
|
* Second step of SSA transform: Introduces new SSA variables at each control-flow join
|
|
|
|
* and at the beginning of functions.
|
|
|
|
*/
|
|
|
|
class IntroduceControlFlowSSA: public ASTModifier
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit IntroduceControlFlowSSA(
|
|
|
|
NameDispenser& _nameDispenser,
|
2020-02-19 16:34:08 +00:00
|
|
|
set<YulString> const& _variablesToReplace,
|
|
|
|
TypeInfo const& _typeInfo
|
2019-09-07 19:14:25 +00:00
|
|
|
):
|
2020-02-19 16:34:08 +00:00
|
|
|
m_nameDispenser(_nameDispenser),
|
|
|
|
m_variablesToReplace(_variablesToReplace),
|
|
|
|
m_typeInfo(_typeInfo)
|
2019-09-07 19:14:25 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
void operator()(FunctionDefinition& _function) override;
|
|
|
|
void operator()(ForLoop& _forLoop) override;
|
|
|
|
void operator()(Switch& _switch) override;
|
|
|
|
void operator()(Block& _block) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
NameDispenser& m_nameDispenser;
|
|
|
|
set<YulString> const& m_variablesToReplace;
|
|
|
|
/// Variables (that are to be replaced) currently in scope.
|
|
|
|
set<YulString> m_variablesInScope;
|
|
|
|
/// Set of variables that do not have a specific value.
|
|
|
|
set<YulString> m_variablesToReassign;
|
2020-02-19 16:34:08 +00:00
|
|
|
TypeInfo const& m_typeInfo;
|
2019-09-07 19:14:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void IntroduceControlFlowSSA::operator()(FunctionDefinition& _function)
|
|
|
|
{
|
|
|
|
set<YulString> varsInScope;
|
|
|
|
std::swap(varsInScope, m_variablesInScope);
|
|
|
|
set<YulString> toReassign;
|
|
|
|
std::swap(toReassign, m_variablesToReassign);
|
|
|
|
|
|
|
|
for (auto const& param: _function.parameters)
|
|
|
|
if (m_variablesToReplace.count(param.name))
|
|
|
|
{
|
|
|
|
m_variablesInScope.insert(param.name);
|
|
|
|
m_variablesToReassign.insert(param.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTModifier::operator()(_function);
|
|
|
|
|
|
|
|
m_variablesInScope = std::move(varsInScope);
|
|
|
|
m_variablesToReassign = std::move(toReassign);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IntroduceControlFlowSSA::operator()(ForLoop& _for)
|
|
|
|
{
|
2020-07-02 15:02:26 +00:00
|
|
|
yulAssert(_for.pre.statements.empty(), "For loop init rewriter not run.");
|
2019-09-07 19:14:25 +00:00
|
|
|
|
|
|
|
Assignments assignments;
|
|
|
|
assignments(_for.body);
|
|
|
|
assignments(_for.post);
|
|
|
|
|
|
|
|
|
|
|
|
for (auto const& var: assignments.names())
|
|
|
|
if (m_variablesInScope.count(var))
|
|
|
|
m_variablesToReassign.insert(var);
|
|
|
|
|
|
|
|
(*this)(_for.body);
|
|
|
|
(*this)(_for.post);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IntroduceControlFlowSSA::operator()(Switch& _switch)
|
|
|
|
{
|
|
|
|
yulAssert(m_variablesToReassign.empty(), "");
|
|
|
|
|
|
|
|
set<YulString> toReassign;
|
|
|
|
for (auto& c: _switch.cases)
|
|
|
|
{
|
|
|
|
(*this)(c.body);
|
|
|
|
toReassign += m_variablesToReassign;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_variablesToReassign += toReassign;
|
|
|
|
}
|
|
|
|
|
|
|
|
void IntroduceControlFlowSSA::operator()(Block& _block)
|
|
|
|
{
|
|
|
|
set<YulString> variablesDeclaredHere;
|
|
|
|
set<YulString> assignedVariables;
|
|
|
|
|
2019-12-11 16:31:36 +00:00
|
|
|
util::iterateReplacing(
|
2019-09-07 19:14:25 +00:00
|
|
|
_block.statements,
|
2019-10-28 10:39:30 +00:00
|
|
|
[&](Statement& _s) -> std::optional<vector<Statement>>
|
2019-09-07 19:14:25 +00:00
|
|
|
{
|
|
|
|
vector<Statement> toPrepend;
|
|
|
|
for (YulString toReassign: m_variablesToReassign)
|
|
|
|
{
|
|
|
|
YulString newName = m_nameDispenser.newName(toReassign);
|
|
|
|
toPrepend.emplace_back(VariableDeclaration{
|
2021-04-27 14:53:04 +00:00
|
|
|
debugDataOf(_s),
|
|
|
|
{TypedName{debugDataOf(_s), newName, m_typeInfo.typeOfVariable(toReassign)}},
|
|
|
|
make_unique<Expression>(Identifier{debugDataOf(_s), toReassign})
|
2019-09-07 19:14:25 +00:00
|
|
|
});
|
|
|
|
assignedVariables.insert(toReassign);
|
|
|
|
}
|
|
|
|
m_variablesToReassign.clear();
|
|
|
|
|
2019-11-19 15:42:49 +00:00
|
|
|
if (holds_alternative<VariableDeclaration>(_s))
|
2019-09-07 19:14:25 +00:00
|
|
|
{
|
2019-11-19 15:42:49 +00:00
|
|
|
VariableDeclaration& varDecl = std::get<VariableDeclaration>(_s);
|
2019-09-07 19:14:25 +00:00
|
|
|
for (auto const& var: varDecl.variables)
|
|
|
|
if (m_variablesToReplace.count(var.name))
|
|
|
|
{
|
|
|
|
variablesDeclaredHere.insert(var.name);
|
|
|
|
m_variablesInScope.insert(var.name);
|
|
|
|
}
|
|
|
|
}
|
2019-11-19 15:42:49 +00:00
|
|
|
else if (holds_alternative<Assignment>(_s))
|
2019-09-07 19:14:25 +00:00
|
|
|
{
|
2019-11-19 15:42:49 +00:00
|
|
|
Assignment& assignment = std::get<Assignment>(_s);
|
2019-09-07 19:14:25 +00:00
|
|
|
for (auto const& var: assignment.variableNames)
|
|
|
|
if (m_variablesToReplace.count(var.name))
|
|
|
|
assignedVariables.insert(var.name);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
visit(_s);
|
|
|
|
|
|
|
|
if (toPrepend.empty())
|
|
|
|
return {};
|
|
|
|
else
|
|
|
|
{
|
|
|
|
toPrepend.emplace_back(std::move(_s));
|
2019-10-28 10:39:30 +00:00
|
|
|
return {std::move(toPrepend)};
|
2019-09-07 19:14:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
m_variablesToReassign += assignedVariables;
|
|
|
|
m_variablesInScope -= variablesDeclaredHere;
|
|
|
|
m_variablesToReassign -= variablesDeclaredHere;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Third step of SSA transform: Replace the references to variables-to-be-replaced
|
2019-09-12 13:19:11 +00:00
|
|
|
* by their current values.
|
|
|
|
*/
|
|
|
|
class PropagateValues: public ASTModifier
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit PropagateValues(set<YulString> const& _variablesToReplace):
|
|
|
|
m_variablesToReplace(_variablesToReplace)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
void operator()(Identifier& _identifier) override;
|
|
|
|
void operator()(VariableDeclaration& _varDecl) override;
|
|
|
|
void operator()(Assignment& _assignment) override;
|
|
|
|
void operator()(ForLoop& _for) override;
|
|
|
|
void operator()(Block& _block) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// This is a set of all variables that are assigned to anywhere in the code.
|
|
|
|
/// Variables that are only declared but never re-assigned are not touched.
|
|
|
|
set<YulString> const& m_variablesToReplace;
|
|
|
|
map<YulString, YulString> m_currentVariableValues;
|
|
|
|
set<YulString> m_clearAtEndOfBlock;
|
|
|
|
};
|
|
|
|
|
|
|
|
void PropagateValues::operator()(Identifier& _identifier)
|
|
|
|
{
|
|
|
|
if (m_currentVariableValues.count(_identifier.name))
|
|
|
|
_identifier.name = m_currentVariableValues[_identifier.name];
|
|
|
|
}
|
|
|
|
|
|
|
|
void PropagateValues::operator()(VariableDeclaration& _varDecl)
|
|
|
|
{
|
|
|
|
ASTModifier::operator()(_varDecl);
|
|
|
|
|
|
|
|
if (_varDecl.variables.size() != 1)
|
|
|
|
return;
|
|
|
|
|
2019-09-07 19:14:25 +00:00
|
|
|
YulString variable = _varDecl.variables.front().name;
|
|
|
|
if (m_variablesToReplace.count(variable))
|
|
|
|
{
|
|
|
|
// `let a := a_1` - regular declaration of non-SSA variable
|
2019-11-19 15:42:49 +00:00
|
|
|
yulAssert(holds_alternative<Identifier>(*_varDecl.value), "");
|
|
|
|
m_currentVariableValues[variable] = std::get<Identifier>(*_varDecl.value).name;
|
2019-09-07 19:14:25 +00:00
|
|
|
m_clearAtEndOfBlock.insert(variable);
|
|
|
|
}
|
2019-11-19 15:42:49 +00:00
|
|
|
else if (_varDecl.value && holds_alternative<Identifier>(*_varDecl.value))
|
2019-09-07 19:14:25 +00:00
|
|
|
{
|
|
|
|
// `let a_1 := a` - assignment to SSA variable after a branch.
|
2019-11-19 15:42:49 +00:00
|
|
|
YulString value = std::get<Identifier>(*_varDecl.value).name;
|
2019-09-07 19:14:25 +00:00
|
|
|
if (m_variablesToReplace.count(value))
|
|
|
|
{
|
|
|
|
// This is safe because `a_1` is not a "variable to replace" and thus
|
|
|
|
// will not be re-assigned.
|
|
|
|
m_currentVariableValues[value] = variable;
|
|
|
|
m_clearAtEndOfBlock.insert(value);
|
|
|
|
}
|
|
|
|
}
|
2019-09-12 13:19:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PropagateValues::operator()(Assignment& _assignment)
|
|
|
|
{
|
|
|
|
visit(*_assignment.value);
|
|
|
|
|
|
|
|
if (_assignment.variableNames.size() != 1)
|
|
|
|
return;
|
|
|
|
YulString name = _assignment.variableNames.front().name;
|
|
|
|
if (!m_variablesToReplace.count(name))
|
|
|
|
return;
|
|
|
|
|
2019-11-19 15:42:49 +00:00
|
|
|
yulAssert(_assignment.value && holds_alternative<Identifier>(*_assignment.value), "");
|
|
|
|
m_currentVariableValues[name] = std::get<Identifier>(*_assignment.value).name;
|
2019-09-12 13:19:11 +00:00
|
|
|
m_clearAtEndOfBlock.insert(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PropagateValues::operator()(ForLoop& _for)
|
|
|
|
{
|
2020-07-02 15:02:26 +00:00
|
|
|
yulAssert(_for.pre.statements.empty(), "For loop init rewriter not run.");
|
2019-09-12 13:19:11 +00:00
|
|
|
|
|
|
|
Assignments assignments;
|
|
|
|
assignments(_for.body);
|
|
|
|
assignments(_for.post);
|
|
|
|
|
|
|
|
for (auto const& var: assignments.names())
|
2018-10-16 21:47:02 +00:00
|
|
|
m_currentVariableValues.erase(var);
|
2019-09-12 13:19:11 +00:00
|
|
|
|
|
|
|
visit(*_for.condition);
|
|
|
|
(*this)(_for.body);
|
|
|
|
(*this)(_for.post);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PropagateValues::operator()(Block& _block)
|
|
|
|
{
|
|
|
|
set<YulString> clearAtParentBlock = std::move(m_clearAtEndOfBlock);
|
|
|
|
m_clearAtEndOfBlock.clear();
|
|
|
|
|
|
|
|
ASTModifier::operator()(_block);
|
|
|
|
|
|
|
|
for (auto const& var: m_clearAtEndOfBlock)
|
|
|
|
m_currentVariableValues.erase(var);
|
|
|
|
|
|
|
|
m_clearAtEndOfBlock = std::move(clearAtParentBlock);
|
|
|
|
}
|
|
|
|
|
2018-10-16 21:47:02 +00:00
|
|
|
}
|
|
|
|
|
2019-09-23 14:32:50 +00:00
|
|
|
void SSATransform::run(OptimiserStepContext& _context, Block& _ast)
|
2018-10-16 21:47:02 +00:00
|
|
|
{
|
2020-02-19 16:34:08 +00:00
|
|
|
TypeInfo typeInfo(_context.dialect, _ast);
|
2018-10-16 21:47:02 +00:00
|
|
|
Assignments assignments;
|
|
|
|
assignments(_ast);
|
2020-02-19 16:34:08 +00:00
|
|
|
IntroduceSSA{_context.dispenser, assignments.names(), typeInfo}(_ast);
|
|
|
|
IntroduceControlFlowSSA{_context.dispenser, assignments.names(), typeInfo}(_ast);
|
2019-09-12 13:19:11 +00:00
|
|
|
PropagateValues{assignments.names()}(_ast);
|
2018-10-16 21:47:02 +00:00
|
|
|
}
|
|
|
|
|
2019-09-12 13:19:11 +00:00
|
|
|
|