/* * Copyright 2015 WebAssembly Community Group participants * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef wasm_pass_h #define wasm_pass_h #include #include "wasm.h" #include "wasm-traversal.h" #include "mixed_arena.h" #include "support/utilities.h" namespace wasm { class Pass; // // Global registry of all passes in /passes/ // struct PassRegistry { PassRegistry(); static PassRegistry* get(); typedef std::function Creator; void registerPass(const char* name, const char *description, Creator create); Pass* createPass(std::string name); std::vector getRegisteredNames(); std::string getPassDescription(std::string name); private: void registerPasses(); struct PassInfo { std::string description; Creator create; PassInfo() {} PassInfo(std::string description, Creator create) : description(description), create(create) {} }; std::map passInfos; }; struct PassOptions { bool debug = false; // run passes in debug mode, doing extra validation and timing checks bool validate = true; // whether to run the validator to check for errors bool validateGlobally = false; // when validating validate globally and not just locally int optimizeLevel = 0; // 0, 1, 2 correspond to -O0, -O1, -O2, etc. int shrinkLevel = 0; // 0, 1, 2 correspond to -O0, -Os, -Oz bool ignoreImplicitTraps = false; // optimize assuming things like div by 0, bad load/store, will not trap bool debugInfo = false; // whether to try to preserve debug info through, which are special calls FeatureSet features = Feature::MVP; // Which wasm features to accept, and be allowed to use void setDefaultOptimizationOptions() { // -Os is our default optimizeLevel = 2; shrinkLevel = 1; } static PassOptions getWithDefaultOptimizationOptions() { PassOptions ret; ret.setDefaultOptimizationOptions(); return ret; } static PassOptions getWithoutOptimization() { return PassOptions(); // defaults are to not optimize } }; // // Runs a set of passes, in order // struct PassRunner { Module* wasm; MixedArena* allocator; std::vector passes; PassOptions options; PassRunner(Module* wasm) : wasm(wasm), allocator(&wasm->allocator) {} PassRunner(Module* wasm, PassOptions options) : wasm(wasm), allocator(&wasm->allocator), options(options) {} // no copying, we control |passes| PassRunner(const PassRunner&) = delete; PassRunner& operator=(const PassRunner&) = delete; void setDebug(bool debug) { options.debug = debug; options.validateGlobally = debug; // validate everything by default if debugging } void setDebugInfo(bool debugInfo) { options.debugInfo = debugInfo; } void setValidateGlobally(bool validate) { options.validateGlobally = validate; } void setFeatures(FeatureSet features) { options.features = features; } void add(std::string passName) { auto pass = PassRegistry::get()->createPass(passName); if (!pass) Fatal() << "Could not find pass: " << passName << "\n"; doAdd(pass); } template void add() { doAdd(new P()); } template void add(Arg arg){ doAdd(new P(arg)); } // Adds the default set of optimization passes; this is // what -O does. void addDefaultOptimizationPasses(); // Adds the default optimization passes that work on // individual functions. void addDefaultFunctionOptimizationPasses(); // Adds the default optimization passes that work on // entire modules as a whole, and make sense to // run before function passes. void addDefaultGlobalOptimizationPrePasses(); // Adds the default optimization passes that work on // entire modules as a whole, and make sense to // run after function passes. // This is run at the very end of the optimization // process - you can assume no other opts will be run // afterwards. void addDefaultGlobalOptimizationPostPasses(); // Run the passes on the module void run(); // Run the passes on a specific function void runOnFunction(Function* func); // Get the last pass that was already executed of a certain type. template P* getLast(); ~PassRunner(); // When running a pass runner within another pass runner, this // flag should be set. This influences how pass debugging works, // and may influence other things in the future too. void setIsNested(bool nested) { isNested = nested; } // BINARYEN_PASS_DEBUG is a convenient commandline way to log out the toplevel passes, their times, // and validate between each pass. // (we don't recurse pass debug into sub-passes, as it doesn't help anyhow and // also is bad for e.g. printing which is a pass) // this method returns whether we are in passDebug mode, and which value: // 1: run pass by pass, validating in between // 2: also save the last pass, so it breakage happens we can print the last one // 3: also dump out byn-* files for each pass static int getPassDebug(); protected: bool isNested = false; private: void doAdd(Pass* pass); void runPass(Pass* pass); void runPassOnFunction(Pass* pass, Function* func); // After running a pass, handle any changes due to // how the pass is defined, such as clearing away any // temporary data structures that the pass declares it // invalidates. // If a function is passed, we operate just on that function; // otherwise, the whole module. void handleAfterEffects(Pass* pass, Function* func=nullptr); }; // // Core pass class // class Pass { public: virtual ~Pass() {}; // Override this to perform preparation work before the pass runs. // This will be called before the pass is run on a module. virtual void prepareToRun(PassRunner* runner, Module* module) {} // Implement this with code to run the pass on the whole module virtual void run(PassRunner* runner, Module* module) { WASM_UNREACHABLE(); } // Implement this with code to run the pass on a single function, for // a function-parallel pass virtual void runOnFunction(PassRunner* runner, Module* module, Function* function) { WASM_UNREACHABLE(); } // Function parallelism. By default, passes are not run in parallel, but you // can override this method to say that functions are parallelizable. This // should always be safe *unless* you do something in the pass that makes it // not thread-safe; in other words, the Module and Function objects and // so forth are set up so that Functions can be processed in parallel, so // if you do not ad global state that could be raced on, your pass could be // function-parallel. // // Function-parallel passes create an instance of the Walker class per function. // That means that you can't rely on Walker object properties to persist across // your functions, and you can't expect a new object to be created for each // function either (which could be very inefficient). // // It is valid for function-parallel passes to read (but not modify) global // module state, like globals or imports. However, reading other functions' // contents is invalid, as function-parallel tests can be run while still // adding functions to the module. virtual bool isFunctionParallel() { return false; } // This method is used to create instances per function for a function-parallel // pass. You may need to override this if you subclass a Walker, as otherwise // this will create the parent class. virtual Pass* create() { WASM_UNREACHABLE(); } // Whether this pass modifies the Binaryen IR in the module. This is true for // most passes, except for passes that have no side effects, or passes that // only modify other things than Binaryen IR (for example, the Stack IR // passes only modify that IR). // This property is important as if Binaryen IR is modified, we need to throw // out any Stack IR - it would need to be regenerated and optimized. virtual bool modifiesBinaryenIR() { return true; } std::string name; protected: Pass() {} Pass(Pass &) {} Pass &operator=(const Pass&) = delete; }; // // Core pass class that uses AST walking. This class can be parameterized by // different types of AST walkers. // template class WalkerPass : public Pass, public WalkerType { PassRunner *runner; protected: typedef WalkerPass super; public: void run(PassRunner* runner, Module* module) override { setPassRunner(runner); WalkerType::setModule(module); WalkerType::walkModule(module); } void runOnFunction(PassRunner* runner, Module* module, Function* func) override { setPassRunner(runner); WalkerType::setModule(module); WalkerType::walkFunction(func); } PassRunner* getPassRunner() { return runner; } PassOptions& getPassOptions() { return runner->options; } void setPassRunner(PassRunner* runner_) { runner = runner_; } }; } // namespace wasm #endif // wasm_pass_h