// Copyright (c) 2016 Google Inc. // // 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. #include "source/opt/pass_manager.h" #include #include #include #include "source/opt/ir_context.h" #include "source/util/timer.h" #include "spirv-tools/libspirv.hpp" namespace spvtools { namespace opt { Pass::Status PassManager::Run(IRContext* context) { auto status = Pass::Status::SuccessWithoutChange; // If print_all_stream_ is not null, prints the disassembly of the module // to that stream, with the given preamble and optionally the pass name. auto print_disassembly = [&context, this](const char* preamble, Pass* pass) { if (print_all_stream_) { std::vector binary; context->module()->ToBinary(&binary, false); SpirvTools t(SPV_ENV_UNIVERSAL_1_2); std::string disassembly; t.Disassemble(binary, &disassembly, 0); *print_all_stream_ << preamble << (pass ? pass->name() : "") << "\n" << disassembly << std::endl; } }; SPIRV_TIMER_DESCRIPTION(time_report_stream_, /* measure_mem_usage = */ true); for (auto& pass : passes_) { print_disassembly("; IR before pass ", pass.get()); SPIRV_TIMER_SCOPED(time_report_stream_, (pass ? pass->name() : ""), true); const auto one_status = pass->Run(context); if (one_status == Pass::Status::Failure) return one_status; if (one_status == Pass::Status::SuccessWithChange) status = one_status; if (validate_after_all_) { spvtools::SpirvTools tools(target_env_); tools.SetMessageConsumer(consumer()); std::vector binary; context->module()->ToBinary(&binary, true); if (!tools.Validate(binary.data(), binary.size(), val_options_)) { std::string msg = "Validation failed after pass "; msg += pass->name(); spv_position_t null_pos{0, 0, 0}; consumer()(SPV_MSG_INTERNAL_ERROR, "", null_pos, msg.c_str()); return Pass::Status::Failure; } } // Reset the pass to free any memory used by the pass. pass.reset(nullptr); } print_disassembly("; IR after last pass", nullptr); // Set the Id bound in the header in case a pass forgot to do so. // // TODO(dnovillo): This should be unnecessary and automatically maintained by // the IRContext. if (status == Pass::Status::SuccessWithChange) { context->module()->SetIdBound(context->module()->ComputeIdBound()); } passes_.clear(); return status; } } // namespace opt } // namespace spvtools