2021-05-05 16:02:35 +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/>.
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* Unit tests for KnowledgeBase
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <test/Common.h>
|
|
|
|
|
|
|
|
#include <test/libyul/Common.h>
|
|
|
|
|
|
|
|
#include <libyul/Object.h>
|
|
|
|
#include <libyul/optimiser/KnowledgeBase.h>
|
|
|
|
#include <libyul/optimiser/SSAValueTracker.h>
|
2021-08-18 16:28:13 +00:00
|
|
|
#include <libyul/optimiser/NameDispenser.h>
|
|
|
|
#include <libyul/optimiser/CommonSubexpressionEliminator.h>
|
2021-05-05 16:02:35 +00:00
|
|
|
#include <libyul/backends/evm/EVMDialect.h>
|
|
|
|
|
|
|
|
#include <liblangutil/ErrorReporter.h>
|
|
|
|
|
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace solidity::langutil;
|
|
|
|
|
|
|
|
namespace solidity::yul::test
|
|
|
|
{
|
|
|
|
|
|
|
|
class KnowledgeBaseTest
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
KnowledgeBase constructKnowledgeBase(string const& _source)
|
|
|
|
{
|
|
|
|
ErrorList errorList;
|
|
|
|
shared_ptr<AsmAnalysisInfo> analysisInfo;
|
2021-08-18 16:28:13 +00:00
|
|
|
std::tie(m_object, analysisInfo) = yul::test::parse(_source, m_dialect, errorList);
|
|
|
|
BOOST_REQUIRE(m_object && errorList.empty() && m_object->code);
|
2021-05-05 16:02:35 +00:00
|
|
|
|
2021-08-18 16:28:13 +00:00
|
|
|
NameDispenser dispenser(m_dialect, *m_object->code);
|
|
|
|
std::set<YulString> reserved;
|
|
|
|
OptimiserStepContext context{m_dialect, dispenser, reserved, 0};
|
|
|
|
CommonSubexpressionEliminator::run(context, *m_object->code);
|
|
|
|
|
|
|
|
m_ssaValues(*m_object->code);
|
|
|
|
for (auto const& [name, expression]: m_ssaValues.values())
|
2021-05-05 16:02:35 +00:00
|
|
|
m_values[name].value = expression;
|
|
|
|
|
2022-11-10 10:49:40 +00:00
|
|
|
return KnowledgeBase([this](YulString _var) { return util::valueOrNullptr(m_values, _var); });
|
2021-05-05 16:02:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
EVMDialect m_dialect{EVMVersion{}, true};
|
2021-08-18 16:28:13 +00:00
|
|
|
shared_ptr<Object> m_object;
|
|
|
|
SSAValueTracker m_ssaValues;
|
2021-05-05 16:02:35 +00:00
|
|
|
map<YulString, AssignedValue> m_values;
|
|
|
|
};
|
|
|
|
|
|
|
|
BOOST_FIXTURE_TEST_SUITE(KnowledgeBase, KnowledgeBaseTest)
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(basic)
|
|
|
|
{
|
|
|
|
yul::KnowledgeBase kb = constructKnowledgeBase(R"({
|
|
|
|
let a := calldataload(0)
|
|
|
|
let b := calldataload(0)
|
|
|
|
let zero := 0
|
|
|
|
let c := add(b, 0)
|
|
|
|
let d := mul(b, 0)
|
|
|
|
let e := sub(a, b)
|
|
|
|
})");
|
|
|
|
|
|
|
|
BOOST_CHECK(!kb.knownToBeDifferent("a"_yulstring, "b"_yulstring));
|
2021-08-18 16:28:13 +00:00
|
|
|
// This only works if the variable names are the same.
|
|
|
|
// It assumes that SSA+CSE+Simplifier actually replaces the variables.
|
|
|
|
BOOST_CHECK(!kb.valueIfKnownConstant("a"_yulstring));
|
2021-05-05 16:02:35 +00:00
|
|
|
BOOST_CHECK(kb.valueIfKnownConstant("zero"_yulstring) == u256(0));
|
2022-11-10 10:49:40 +00:00
|
|
|
BOOST_CHECK(kb.differenceIfKnownConstant("a"_yulstring, "b"_yulstring) == u256(0));
|
|
|
|
BOOST_CHECK(kb.differenceIfKnownConstant("a"_yulstring, "c"_yulstring) == u256(0));
|
|
|
|
BOOST_CHECK(kb.valueIfKnownConstant("e"_yulstring) == u256(0));
|
2021-08-18 16:28:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(difference)
|
|
|
|
{
|
|
|
|
yul::KnowledgeBase kb = constructKnowledgeBase(R"({
|
|
|
|
let a := calldataload(0)
|
|
|
|
let b := add(a, 200)
|
|
|
|
let c := add(a, 220)
|
2022-11-10 10:49:40 +00:00
|
|
|
let d := add(12, c)
|
2021-08-18 16:28:13 +00:00
|
|
|
let e := sub(c, 12)
|
|
|
|
})");
|
2021-05-05 16:02:35 +00:00
|
|
|
|
2021-08-18 16:28:13 +00:00
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("c"_yulstring, "b"_yulstring) ==
|
|
|
|
u256(20)
|
|
|
|
);
|
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("b"_yulstring, "c"_yulstring) ==
|
|
|
|
u256(-20)
|
|
|
|
);
|
|
|
|
BOOST_CHECK(!kb.knownToBeDifferentByAtLeast32("b"_yulstring, "c"_yulstring));
|
|
|
|
BOOST_CHECK(kb.knownToBeDifferentByAtLeast32("b"_yulstring, "d"_yulstring));
|
|
|
|
BOOST_CHECK(kb.knownToBeDifferentByAtLeast32("a"_yulstring, "b"_yulstring));
|
|
|
|
BOOST_CHECK(kb.knownToBeDifferentByAtLeast32("b"_yulstring, "a"_yulstring));
|
|
|
|
|
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("e"_yulstring, "a"_yulstring) == u256(208)
|
|
|
|
);
|
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("e"_yulstring, "b"_yulstring) == u256(8)
|
|
|
|
);
|
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("a"_yulstring, "e"_yulstring) == u256(-208)
|
|
|
|
);
|
|
|
|
BOOST_CHECK(
|
|
|
|
kb.differenceIfKnownConstant("b"_yulstring, "e"_yulstring) == u256(-8)
|
|
|
|
);
|
2021-05-05 16:02:35 +00:00
|
|
|
}
|
|
|
|
|
2021-08-18 16:28:13 +00:00
|
|
|
|
2021-05-05 16:02:35 +00:00
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|
|
|
|
|
|
|
|
}
|