|  | // 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 <iostream> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #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<uint32_t> 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<uint32_t> 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 |