/* 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 . */ /** * Unit tests for KnowledgeBase */ #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace solidity::langutil; namespace solidity::yul::test { class KnowledgeBaseTest { protected: KnowledgeBase constructKnowledgeBase(string const& _source) { ErrorList errorList; shared_ptr analysisInfo; std::tie(m_object, analysisInfo) = yul::test::parse(_source, m_dialect, errorList); BOOST_REQUIRE(m_object && errorList.empty() && m_object->code); NameDispenser dispenser(m_dialect, *m_object->code); std::set 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()) m_values[name].value = expression; return KnowledgeBase([this](YulString _var) { return util::valueOrNullptr(m_values, _var); }); } EVMDialect m_dialect{EVMVersion{}, true}; shared_ptr m_object; SSAValueTracker m_ssaValues; map 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)); // 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)); BOOST_CHECK(kb.valueIfKnownConstant("zero"_yulstring) == u256(0)); 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)); } BOOST_AUTO_TEST_CASE(difference) { yul::KnowledgeBase kb = constructKnowledgeBase(R"({ let a := calldataload(0) let b := add(a, 200) let c := add(a, 220) let d := add(12, c) let e := sub(c, 12) })"); 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) ); } BOOST_AUTO_TEST_SUITE_END() }