/* 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 . */ /** * 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 #include #include #include #include #include #include #include #include #include using namespace std; using namespace dev; using namespace langutil; using namespace dev::solidity; using namespace yul; namespace po = boost::program_options; class YulOpti { public: void printErrors() { SourceReferenceFormatter formatter(cout); for (auto const& error: m_errors) formatter.printErrorInformation(*error); } bool parse(string const& _input) { ErrorReporter errorReporter(m_errors); shared_ptr scanner = make_shared(CharStream(_input, "")); m_ast = yul::Parser(errorReporter, m_dialect).parse(scanner, false); if (!m_ast || !errorReporter.errors().empty()) { cout << "Error parsing source." << endl; printErrors(); return false; } m_analysisInfo = make_shared(); AsmAnalyzer analyzer( *m_analysisInfo, errorReporter, m_dialect ); if (!analyzer.analyze(*m_ast) || !errorReporter.errors().empty()) { cout << "Error analyzing source." << endl; printErrors(); return false; } return true; } void runInteractive(string source) { bool disambiguated = false; while (true) { cout << "----------------------" << endl; cout << source << endl; if (!parse(source)) return; set reservedIdentifiers; if (!disambiguated) { *m_ast = boost::get(Disambiguator(m_dialect, *m_analysisInfo)(*m_ast)); m_analysisInfo.reset(); m_nameDispenser = make_shared(m_dialect, *m_ast, reservedIdentifiers); disambiguated = true; } cout << "(q)quit/(f)flatten/(c)se/initialize var(d)ecls/(x)plit/(j)oin/(g)rouper/(h)oister/" << endl; cout << " (e)xpr inline/(i)nline/(s)implify/varname c(l)eaner/(u)nusedprune/ss(a) transform/" << endl; cout << " (r)edundant assign elim./re(m)aterializer/f(o)r-loop-init-rewriter/for-loop-condition-(I)nto-body/" << endl; cout << " for-loop-condition-(O)ut-of-body/s(t)ructural simplifier/equi(v)alent function combiner/ssa re(V)erser/" << endl; cout << " co(n)trol flow simplifier/stack com(p)ressor/(D)ead code eliminator/(L)oad resolver/? " << endl; cout.flush(); int option = readStandardInputChar(); cout << ' ' << char(option) << endl; OptimiserStepContext context{m_dialect, *m_nameDispenser, reservedIdentifiers}; switch (option) { case 'q': return; case 'f': BlockFlattener::run(context, *m_ast); break; case 'o': ForLoopInitRewriter::run(context, *m_ast); break; case 'O': ForLoopConditionOutOfBody::run(context, *m_ast); break; case 'I': ForLoopConditionIntoBody::run(context, *m_ast); break; case 'c': CommonSubexpressionEliminator::run(context, *m_ast); break; case 'd': VarDeclInitializer::run(context, *m_ast); break; case 'l': VarNameCleaner::run(context, *m_ast); break; case 'x': ExpressionSplitter::run(context, *m_ast); break; case 'j': ExpressionJoiner::run(context, *m_ast); break; case 'g': FunctionGrouper::run(context, *m_ast); break; case 'h': FunctionHoister::run(context, *m_ast); break; case 'e': ExpressionInliner::run(context, *m_ast); break; case 'i': FullInliner::run(context, *m_ast); break; case 's': ExpressionSimplifier::run(context, *m_ast); break; case 't': StructuralSimplifier::run(context, *m_ast); break; case 'T': LiteralRematerialiser::run(context, *m_ast); break; case 'n': ControlFlowSimplifier::run(context, *m_ast); break; case 'u': UnusedPruner::run(context, *m_ast); break; case 'D': DeadCodeEliminator::run(context, *m_ast); break; case 'a': SSATransform::run(context, *m_ast); break; case 'r': RedundantAssignEliminator::run(context, *m_ast); break; case 'm': Rematerialiser::run(context, *m_ast); break; case 'v': EquivalentFunctionCombiner::run(context, *m_ast); break; case 'V': SSAReverser::run(context, *m_ast); break; case 'p': { Object obj; obj.code = m_ast; StackCompressor::run(m_dialect, obj, true, 16); break; } case 'L': LoadResolver::run(context, *m_ast); break; default: cout << "Unknown option." << endl; } source = AsmPrinter{}(*m_ast); } } private: ErrorList m_errors; shared_ptr m_ast; Dialect const& m_dialect{EVMDialect::strictAssemblyForEVMObjects(EVMVersion{})}; shared_ptr m_analysisInfo; shared_ptr m_nameDispenser; }; int main(int argc, char** argv) { po::options_description options( R"(yulopti, yul optimizer exploration tool. Usage: yulopti [Options] Reads as yul code and applies optimizer steps to it, interactively read from stdin. 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" ) ("help", "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; try { po::command_line_parser cmdLineParser(argc, argv); cmdLineParser.options(options).positional(filesPositions); po::store(cmdLineParser.run(), arguments); } catch (po::error const& _exception) { cerr << _exception.what() << endl; return 1; } string input; if (arguments.count("input-file")) YulOpti{}.runInteractive(readFileAsString(arguments["input-file"].as())); else cout << options; return 0; }