/* 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 /** * Interactive yul optimizer */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace solidity; using namespace solidity::util; using namespace solidity::langutil; using namespace solidity::frontend; using namespace solidity::yul; namespace po = boost::program_options; class YulOpti { public: static void printErrors(CharStream const& _charStream, ErrorList const& _errors) { SourceReferenceFormatter{ cerr, SingletonCharStreamProvider(_charStream), true, false }.printErrorInformation(_errors); } shared_ptr getSubObject(shared_ptr const& _object, string const& _qualifiedPath) { if (_qualifiedPath.empty() || _qualifiedPath == _object->name.str()) return _object; yulAssert( boost::algorithm::starts_with(_qualifiedPath, _object->name.str() + "."), "Assembly object not found." ); string const subObjectPath = _qualifiedPath.substr(_object->name.str().length() + 1); string const subObjectName = subObjectPath.substr(0, subObjectPath.find_first_of('.')); auto subObjectIt = ranges::find_if( _object->subObjects, [&subObjectName](auto const& _subObject) { return _subObject->name.str() == subObjectName; } ); yulAssert( subObjectIt != _object->subObjects.end(), "Assembly object not found." ); auto subObject = dynamic_pointer_cast(*subObjectIt); yulAssert( subObject != nullptr, "Assembly object may not contain code." ); return getSubObject(subObject, subObjectPath); } void parse(string const& _input, string const& _objectPath) { ErrorList errors; ErrorReporter errorReporter(errors); CharStream charStream(_input, ""); try { ObjectParser parser(errorReporter, m_dialect); auto scanner = make_shared(charStream); if (!m_inputWasCodeBlock && scanner->currentToken() == Token::LBrace) m_inputWasCodeBlock = true; shared_ptr object = parser.parse(scanner, false); if (!object || !errorReporter.errors().empty()) { cerr << "Error parsing source." << endl; printErrors(charStream, errors); throw runtime_error("Could not parse source."); } m_object = getSubObject(object, _objectPath); runCodeAnalyzer(errorReporter); } catch(...) { cerr << "Fatal error during parsing: " << endl; printErrors(charStream, errors); throw; } } void printUsageBanner( map const& _extraOptions, size_t _columns ) { yulAssert(_columns > 0); auto const& optimiserSteps = OptimiserSuite::stepAbbreviationToNameMap(); auto hasShorterString = [](auto const& a, auto const& b) { return a.second.size() < b.second.size(); }; size_t longestDescriptionLength = max( max_element(optimiserSteps.begin(), optimiserSteps.end(), hasShorterString)->second.size(), max_element(_extraOptions.begin(), _extraOptions.end(), hasShorterString)->second.size() ); vector overlappingAbbreviations = ranges::views::set_intersection(_extraOptions | ranges::views::keys, optimiserSteps | ranges::views::keys) | ranges::views::transform([](char _abbreviation){ return string(1, _abbreviation); }) | ranges::to(); yulAssert( overlappingAbbreviations.empty(), "ERROR: Conflict between yulopti controls and the following Yul optimizer step abbreviations: " + boost::join(overlappingAbbreviations, ", ") + ".\n" "This is most likely caused by someone adding a new step abbreviation to " "OptimiserSuite::stepNameToAbbreviationMap() and not realizing that it's used by yulopti.\n" "Please update the code to use a different character and recompile yulopti." ); vector> sortedOptions = ranges::views::concat(optimiserSteps, _extraOptions) | ranges::to>>() | ranges::actions::sort([](tuple const& _a, tuple const& _b) { return ( !boost::algorithm::iequals(get<1>(_a), get<1>(_b)) ? boost::algorithm::lexicographical_compare(get<1>(_a), get<1>(_b), boost::algorithm::is_iless()) : toLower(get<0>(_a)) < toLower(get<0>(_b)) ); }); yulAssert(sortedOptions.size() > 0); size_t rows = (sortedOptions.size() - 1) / _columns + 1; for (size_t row = 0; row < rows; ++row) { for (auto const& [key, name]: sortedOptions | ranges::views::drop(row) | ranges::views::stride(rows)) cout << key << ": " << setw(static_cast(longestDescriptionLength)) << setiosflags(ios::left) << name << " "; cout << endl; } } void applyFunctionToObject(shared_ptr _object, function _function) { for (auto const& subObjectNode: _object->subObjects) { auto subObject = dynamic_pointer_cast(subObjectNode); if (subObject != nullptr) applyFunctionToObject(subObject, _function); } _function(*_object); } void runCodeAnalyzer(ErrorReporter& _errorReporter) { applyFunctionToObject( m_object, [&](Object& _object) { _object.analysisInfo = make_shared(); AsmAnalyzer analyzer( *_object.analysisInfo, _errorReporter, m_dialect, {}, _object.qualifiedDataNames() ); bool success = analyzer.analyze(*_object.code); yulAssert(success && !_errorReporter.hasErrors(), "Invalid assembly/yul code."); } ); } void runCodeDisambiguator() { applyFunctionToObject( m_object, [&](Object& _object) { _object.code = make_shared( get(Disambiguator(m_dialect, *_object.analysisInfo)(*_object.code)) ); _object.analysisInfo.reset(); } ); } void runSequence(string_view _steps) { applyFunctionToObject( m_object, [&](Object& _object) { OptimiserSuite{*m_context}.runSequence(_steps, *_object.code); } ); } void runVarNameCleaner() { applyFunctionToObject( m_object, [&](Object& _object) { VarNameCleaner::run(*m_context, *_object.code); } ); } void runStackCompressor() { applyFunctionToObject( m_object, [&](Object& _object) { StackCompressor::run(m_dialect, _object, true, 16); } ); } void parseAndPrint(string _source, string _objectPath) { parse(_source, _objectPath); printObject(); } void printObject() { if (!m_inputWasCodeBlock) cout << m_object->toString(&m_dialect) << endl; else cout << AsmPrinter{m_dialect}(*m_object->code) << endl; } void resetNameDispenser() { m_nameDispenser = make_shared( m_dialect, m_reservedIdentifiers ); m_context = make_shared( OptimiserStepContext{ m_dialect, *m_nameDispenser, m_reservedIdentifiers, solidity::frontend::OptimiserSettings::standard().expectedExecutionsPerDeployment } ); } void runSteps(string _source, string _objectPath, string _steps) { parse(_source, _objectPath); runCodeDisambiguator(); runSequence(_steps); } void runInteractive(string _source, string const& _objectPath, bool _disambiguated = false) { ErrorList errors; ErrorReporter errorReporter(errors); bool disambiguated = _disambiguated; parse(_source, _objectPath); while (true) { disambiguated = disambiguated || (runCodeDisambiguator(), true); map const& extraOptions = { // QUIT starts with a non-letter character on purpose to get it to show up on top of the list {'#', ">>> QUIT <<<"}, {',', "VarNameCleaner"}, {';', "StackCompressor"} }; printUsageBanner(extraOptions, 4); cout << "? "; cout.flush(); char option = static_cast(readStandardInputChar()); cout << ' ' << option << endl; try { switch (option) { case 4: case '#': return; case ',': runVarNameCleaner(); // VarNameCleaner destroys the unique names guarantee of the disambiguator. disambiguated = false; break; case ';': runStackCompressor(); break; default: runSequence(string_view(&option, 1)); } resetNameDispenser(); runCodeAnalyzer(errorReporter); } catch (...) { cerr << endl << "Exception during optimiser step:" << endl; cerr << boost::current_exception_diagnostic_information() << endl; } cout << "----------------------" << endl; printObject(); } } private: shared_ptr m_object; bool m_inputWasCodeBlock = false; Dialect const& m_dialect{EVMDialect::strictAssemblyForEVMObjects(EVMVersion{})}; set const m_reservedIdentifiers = {}; shared_ptr m_nameDispenser = make_shared( m_dialect, m_reservedIdentifiers ); shared_ptr m_context = make_shared( OptimiserStepContext{ m_dialect, *m_nameDispenser, m_reservedIdentifiers, solidity::frontend::OptimiserSettings::standard().expectedExecutionsPerDeployment } ); }; int main(int argc, char** argv) { try { bool nonInteractive = false; po::options_description options( R"(yulopti, yul optimizer exploration tool. Usage: yulopti [Options] Reads containing a yul object and applies optimizer steps to it, interactively read from stdin. In non-interactive mode a list of steps has to be provided. If is -, yul code is read from stdin and run non-interactively. An flag may be provided, specifying a dotted path to an object in the input. Allowed options)", po::options_description::m_default_line_length, po::options_description::m_default_line_length - 23); options.add_options() ( "input-file", po::value(), "input file" ) ( "steps", po::value(), "steps to execute non-interactively" ) ( "object", po::value(), "path to a yul object in the input" ) ( "non-interactive,n", po::bool_switch(&nonInteractive)->default_value(false), "stop after executing the provided steps" ) ("help,h", "Show this help screen."); // All positional options should be interpreted as input files po::positional_options_description filesPositions; filesPositions.add("input-file", 1); po::variables_map arguments; po::command_line_parser cmdLineParser(argc, argv); cmdLineParser.options(options).positional(filesPositions); po::store(cmdLineParser.run(), arguments); po::notify(arguments); if (arguments.count("help")) { cout << options; return 0; } string input; string objectPath; if (arguments.count("input-file")) { string filename = arguments["input-file"].as(); if (filename == "-") { nonInteractive = true; input = readUntilEnd(cin); } else input = readFileAsString(arguments["input-file"].as()); } else { cout << options; return 1; } if (nonInteractive && !arguments.count("steps")) { cout << options; return 1; } if (arguments.count("object")) objectPath = arguments["object"].as(); YulOpti yulOpti; bool disambiguated = false; if (!nonInteractive) yulOpti.parseAndPrint(input, objectPath); if (arguments.count("steps")) { string sequence = arguments["steps"].as(); if (!nonInteractive) cout << "----------------------" << endl; yulOpti.runSteps(input, objectPath, sequence); disambiguated = true; } if (!nonInteractive) yulOpti.runInteractive(input, objectPath, disambiguated); return 0; } catch (po::error const& _exception) { cerr << _exception.what() << endl; return 1; } catch (FileNotFound const& _exception) { cerr << "File not found:" << _exception.comment() << endl; return 1; } catch (NotAFile const& _exception) { cerr << "Not a regular file:" << _exception.comment() << endl; return 1; } catch(...) { cerr << endl << "Exception:" << endl; cerr << boost::current_exception_diagnostic_information() << endl; return 1; } }