/* 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 . */ // SPDX-License-Identifier: GPL-3.0 /** * @author Lefteris Karapetsas * @date 2015 * Unit tests for Assembly Items from evmasm/Assembly.h */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace solidity::langutil; using namespace solidity::evmasm; namespace solidity::frontend::test { namespace { evmasm::AssemblyItems compileContract(std::shared_ptr _sourceCode) { ErrorList errors; ErrorReporter errorReporter(errors); Parser parser(errorReporter, solidity::test::CommonOptions::get().evmVersion()); ASTPointer sourceUnit; BOOST_REQUIRE_NO_THROW(sourceUnit = parser.parse(*_sourceCode)); BOOST_CHECK(!!sourceUnit); Scoper::assignScopes(*sourceUnit); BOOST_REQUIRE(SyntaxChecker(errorReporter, false).checkSyntax(*sourceUnit)); GlobalContext globalContext; NameAndTypeResolver resolver(globalContext, solidity::test::CommonOptions::get().evmVersion(), errorReporter, false); DeclarationTypeChecker declarationTypeChecker(errorReporter, solidity::test::CommonOptions::get().evmVersion()); solAssert(!Error::containsErrors(errorReporter.errors()), ""); resolver.registerDeclarations(*sourceUnit); BOOST_REQUIRE_NO_THROW(resolver.resolveNamesAndTypes(*sourceUnit)); if (Error::containsErrors(errorReporter.errors())) return AssemblyItems(); for (ASTPointer const& node: sourceUnit->nodes()) { BOOST_REQUIRE_NO_THROW(declarationTypeChecker.check(*node)); if (Error::containsErrors(errorReporter.errors())) return AssemblyItems(); } TypeChecker checker(solidity::test::CommonOptions::get().evmVersion(), errorReporter); BOOST_REQUIRE_NO_THROW(checker.checkTypeRequirements(*sourceUnit)); if (Error::containsErrors(errorReporter.errors())) return AssemblyItems(); for (ASTPointer const& node: sourceUnit->nodes()) if (ContractDefinition* contract = dynamic_cast(node.get())) { Compiler compiler( solidity::test::CommonOptions::get().evmVersion(), RevertStrings::Default, solidity::test::CommonOptions::get().optimize ? OptimiserSettings::standard() : OptimiserSettings::minimal() ); compiler.compileContract(*contract, std::map>{}, bytes()); return compiler.runtimeAssembly().items(); } BOOST_FAIL("No contract found in source."); return AssemblyItems(); } void printAssemblyLocations(AssemblyItems const& _items) { auto printRepeated = [](SourceLocation const& _loc, size_t _repetitions) { std::cout << "\t\tvector(" << _repetitions << ", SourceLocation{" << _loc.start << ", " << _loc.end << ", make_shared(\"" << *_loc.sourceName << "\")}) +" << std::endl; }; std::vector locations; for (auto const& item: _items) locations.push_back(item.location()); size_t repetitions = 0; SourceLocation const* previousLoc = nullptr; for (size_t i = 0; i < locations.size(); ++i) { SourceLocation& loc = locations[i]; if (previousLoc && *previousLoc == loc) repetitions++; else { if (previousLoc) printRepeated(*previousLoc, repetitions); previousLoc = &loc; repetitions = 1; } } if (previousLoc) printRepeated(*previousLoc, repetitions); } void checkAssemblyLocations(AssemblyItems const& _items, std::vector const& _locations) { BOOST_CHECK_EQUAL(_items.size(), _locations.size()); for (size_t i = 0; i < std::min(_items.size(), _locations.size()); ++i) { if (_items[i].location().start != _locations[i].start || _items[i].location().end != _locations[i].end) { BOOST_CHECK_MESSAGE(false, "Location mismatch for item " + std::to_string(i) + ". Found the following locations:"); printAssemblyLocations(_items); return; } } } } // end anonymous namespace BOOST_AUTO_TEST_SUITE(Assembly) BOOST_AUTO_TEST_CASE(location_test) { std::string sourceCode = R"( pragma abicoder v1; contract test { function f() public returns (uint256 a) { return 16; } } )"; AssemblyItems items = compileContract(std::make_shared(sourceCode, "")); std::shared_ptr sourceName = std::make_shared(); bool hasShifts = solidity::test::CommonOptions::get().evmVersion().hasBitwiseShifting(); auto codegenCharStream = std::make_shared("", "--CODEGEN--"); std::vector locations; if (solidity::test::CommonOptions::get().optimize) locations = std::vector(31, SourceLocation{23, 103, sourceName}) + std::vector(1, SourceLocation{41, 100, sourceName}) + std::vector(1, SourceLocation{93, 95, sourceName}) + std::vector(15, SourceLocation{41, 100, sourceName}); else locations = std::vector(hasShifts ? 31 : 32, SourceLocation{23, 103, sourceName}) + std::vector(24, SourceLocation{41, 100, sourceName}) + std::vector(1, SourceLocation{70, 79, sourceName}) + std::vector(1, SourceLocation{93, 95, sourceName}) + std::vector(2, SourceLocation{86, 95, sourceName}) + std::vector(2, SourceLocation{41, 100, sourceName}); checkAssemblyLocations(items, locations); } BOOST_AUTO_TEST_CASE(jump_type) { auto sourceCode = std::make_shared(R"( pragma abicoder v1; contract C { function f(uint a) public pure returns (uint t) { assembly { function g(x) -> y { if x { leave } y := 8 } t := g(a) } } } )", ""); AssemblyItems items = compileContract(sourceCode); std::string jumpTypes; for (AssemblyItem const& item: items) if (item.getJumpType() != AssemblyItem::JumpType::Ordinary) jumpTypes += item.getJumpTypeAsString() + "\n"; if (solidity::test::CommonOptions::get().optimize) BOOST_CHECK_EQUAL(jumpTypes, "[in]\n[out]\n[out]\n[in]\n[out]\n"); else BOOST_CHECK_EQUAL(jumpTypes, "[in]\n[out]\n[in]\n[out]\n"); } BOOST_AUTO_TEST_SUITE_END() } // end namespaces