// Copyright (c) 2020 Google LLC // // 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/debug_info_manager.h" #include #include "source/opt/ir_context.h" // Constants for OpenCL.DebugInfo.100 extension instructions. static const uint32_t kOpLineOperandLineIndex = 1; static const uint32_t kLineOperandIndexDebugFunction = 7; static const uint32_t kLineOperandIndexDebugLexicalBlock = 5; static const uint32_t kDebugFunctionOperandFunctionIndex = 13; static const uint32_t kDebugInlinedAtOperandInlinedIndex = 6; namespace spvtools { namespace opt { namespace analysis { namespace { void SetInlinedOperand(Instruction* dbg_inlined_at, uint32_t inlined_operand) { assert(dbg_inlined_at); assert(dbg_inlined_at->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugInlinedAt); if (dbg_inlined_at->NumOperands() <= kDebugInlinedAtOperandInlinedIndex) { dbg_inlined_at->AddOperand({SPV_OPERAND_TYPE_RESULT_ID, {inlined_operand}}); } else { dbg_inlined_at->SetOperand(kDebugInlinedAtOperandInlinedIndex, {inlined_operand}); } } uint32_t GetInlinedOperand(Instruction* dbg_inlined_at) { assert(dbg_inlined_at); assert(dbg_inlined_at->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugInlinedAt); if (dbg_inlined_at->NumOperands() <= kDebugInlinedAtOperandInlinedIndex) return kNoInlinedAt; return dbg_inlined_at->GetSingleWordOperand( kDebugInlinedAtOperandInlinedIndex); } } // namespace DebugInfoManager::DebugInfoManager(IRContext* c) : context_(c) { AnalyzeDebugInsts(*c->module()); } Instruction* DebugInfoManager::GetDbgInst(uint32_t id) { auto dbg_inst_it = id_to_dbg_inst_.find(id); return dbg_inst_it == id_to_dbg_inst_.end() ? nullptr : dbg_inst_it->second; } void DebugInfoManager::RegisterDbgInst(Instruction* inst) { assert( inst->NumInOperands() != 0 && context()->get_feature_mgr()->GetExtInstImportId_OpenCL100DebugInfo() == inst->GetInOperand(0).words[0] && "Given instruction is not a debug instruction"); id_to_dbg_inst_[inst->result_id()] = inst; } void DebugInfoManager::RegisterDbgFunction(Instruction* inst) { assert(inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugFunction && "inst is not a DebugFunction"); auto fn_id = inst->GetSingleWordOperand(kDebugFunctionOperandFunctionIndex); assert( fn_id_to_dbg_fn_.find(fn_id) == fn_id_to_dbg_fn_.end() && "Register DebugFunction for a function that already has DebugFunction"); fn_id_to_dbg_fn_[fn_id] = inst; } uint32_t DebugInfoManager::CreateDebugInlinedAt(const Instruction* line, const DebugScope& scope) { if (context()->get_feature_mgr()->GetExtInstImportId_OpenCL100DebugInfo() == 0) return kNoInlinedAt; uint32_t line_number = 0; if (line == nullptr) { auto* lexical_scope_inst = GetDbgInst(scope.GetLexicalScope()); if (lexical_scope_inst == nullptr) return kNoInlinedAt; OpenCLDebugInfo100Instructions debug_opcode = lexical_scope_inst->GetOpenCL100DebugOpcode(); switch (debug_opcode) { case OpenCLDebugInfo100DebugFunction: line_number = lexical_scope_inst->GetSingleWordOperand( kLineOperandIndexDebugFunction); break; case OpenCLDebugInfo100DebugLexicalBlock: line_number = lexical_scope_inst->GetSingleWordOperand( kLineOperandIndexDebugLexicalBlock); break; case OpenCLDebugInfo100DebugTypeComposite: case OpenCLDebugInfo100DebugCompilationUnit: assert(false && "DebugTypeComposite and DebugCompilationUnit are lexical " "scopes, but we inline functions into a function or a block " "of a function, not into a struct/class or a global scope."); break; default: assert(false && "Unreachable. a debug extension instruction for a " "lexical scope must be DebugFunction, DebugTypeComposite, " "DebugLexicalBlock, or DebugCompilationUnit."); break; } } else { line_number = line->GetSingleWordOperand(kOpLineOperandLineIndex); } uint32_t result_id = context()->TakeNextId(); std::unique_ptr inlined_at(new Instruction( context(), SpvOpExtInst, context()->get_type_mgr()->GetVoidTypeId(), result_id, { {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {context() ->get_feature_mgr() ->GetExtInstImportId_OpenCL100DebugInfo()}}, {spv_operand_type_t::SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, {static_cast(OpenCLDebugInfo100DebugInlinedAt)}}, {spv_operand_type_t::SPV_OPERAND_TYPE_LITERAL_INTEGER, {line_number}}, {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {scope.GetLexicalScope()}}, })); // |scope| already has DebugInlinedAt. We put the existing DebugInlinedAt // into the Inlined operand of this new DebugInlinedAt. if (scope.GetInlinedAt() != kNoInlinedAt) { inlined_at->AddOperand({spv_operand_type_t::SPV_OPERAND_TYPE_RESULT_ID, {scope.GetInlinedAt()}}); } RegisterDbgInst(inlined_at.get()); context()->module()->AddExtInstDebugInfo(std::move(inlined_at)); return result_id; } DebugScope DebugInfoManager::BuildDebugScope( const DebugScope& callee_instr_scope, DebugInlinedAtContext* inlined_at_ctx) { return DebugScope(callee_instr_scope.GetLexicalScope(), BuildDebugInlinedAtChain(callee_instr_scope.GetInlinedAt(), inlined_at_ctx)); } uint32_t DebugInfoManager::BuildDebugInlinedAtChain( uint32_t callee_inlined_at, DebugInlinedAtContext* inlined_at_ctx) { if (inlined_at_ctx->GetScopeOfCallInstruction().GetLexicalScope() == kNoDebugScope) return kNoInlinedAt; // Reuse the already generated DebugInlinedAt chain if exists. uint32_t already_generated_chain_head_id = inlined_at_ctx->GetDebugInlinedAtChain(callee_inlined_at); if (already_generated_chain_head_id != kNoInlinedAt) { return already_generated_chain_head_id; } const uint32_t new_dbg_inlined_at_id = CreateDebugInlinedAt(inlined_at_ctx->GetLineOfCallInstruction(), inlined_at_ctx->GetScopeOfCallInstruction()); if (new_dbg_inlined_at_id == kNoInlinedAt) return kNoInlinedAt; if (callee_inlined_at == kNoInlinedAt) { inlined_at_ctx->SetDebugInlinedAtChain(kNoInlinedAt, new_dbg_inlined_at_id); return new_dbg_inlined_at_id; } uint32_t chain_head_id = kNoInlinedAt; uint32_t chain_iter_id = callee_inlined_at; Instruction* last_inlined_at_in_chain = nullptr; do { Instruction* new_inlined_at_in_chain = CloneDebugInlinedAt( chain_iter_id, /* insert_before */ last_inlined_at_in_chain); assert(new_inlined_at_in_chain != nullptr); // Set DebugInlinedAt of the new scope as the head of the chain. if (chain_head_id == kNoInlinedAt) chain_head_id = new_inlined_at_in_chain->result_id(); // Previous DebugInlinedAt of the chain must point to the new // DebugInlinedAt as its Inlined operand to build a recursive // chain. if (last_inlined_at_in_chain != nullptr) { SetInlinedOperand(last_inlined_at_in_chain, new_inlined_at_in_chain->result_id()); } last_inlined_at_in_chain = new_inlined_at_in_chain; chain_iter_id = GetInlinedOperand(new_inlined_at_in_chain); } while (chain_iter_id != kNoInlinedAt); // Put |new_dbg_inlined_at_id| into the end of the chain. SetInlinedOperand(last_inlined_at_in_chain, new_dbg_inlined_at_id); // Keep the new chain information that will be reused it. inlined_at_ctx->SetDebugInlinedAtChain(callee_inlined_at, chain_head_id); return chain_head_id; } Instruction* DebugInfoManager::GetDebugInfoNone() { if (debug_info_none_inst_ != nullptr) return debug_info_none_inst_; uint32_t result_id = context()->TakeNextId(); std::unique_ptr dbg_info_none_inst(new Instruction( context(), SpvOpExtInst, context()->get_type_mgr()->GetVoidTypeId(), result_id, { {SPV_OPERAND_TYPE_RESULT_ID, {context() ->get_feature_mgr() ->GetExtInstImportId_OpenCL100DebugInfo()}}, {SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, {static_cast(OpenCLDebugInfo100DebugInfoNone)}}, })); // Add to the front of |ext_inst_debuginfo_|. debug_info_none_inst_ = context()->module()->ext_inst_debuginfo_begin()->InsertBefore( std::move(dbg_info_none_inst)); RegisterDbgInst(debug_info_none_inst_); return debug_info_none_inst_; } Instruction* DebugInfoManager::GetDebugInlinedAt(uint32_t dbg_inlined_at_id) { auto* inlined_at = GetDbgInst(dbg_inlined_at_id); if (inlined_at == nullptr) return nullptr; if (inlined_at->GetOpenCL100DebugOpcode() != OpenCLDebugInfo100DebugInlinedAt) { return nullptr; } return inlined_at; } Instruction* DebugInfoManager::CloneDebugInlinedAt(uint32_t clone_inlined_at_id, Instruction* insert_before) { auto* inlined_at = GetDebugInlinedAt(clone_inlined_at_id); if (inlined_at == nullptr) return nullptr; std::unique_ptr new_inlined_at(inlined_at->Clone(context())); new_inlined_at->SetResultId(context()->TakeNextId()); RegisterDbgInst(new_inlined_at.get()); if (insert_before != nullptr) return insert_before->InsertBefore(std::move(new_inlined_at)); return context()->module()->ext_inst_debuginfo_end()->InsertBefore( std::move(new_inlined_at)); } void DebugInfoManager::AnalyzeDebugInst(Instruction* dbg_inst) { if (dbg_inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100InstructionsMax) return; RegisterDbgInst(dbg_inst); if (dbg_inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugFunction) { assert(GetDebugFunction(dbg_inst->GetSingleWordOperand( kDebugFunctionOperandFunctionIndex)) == nullptr && "Two DebugFunction instruction exists for a single OpFunction."); RegisterDbgFunction(dbg_inst); } if (debug_info_none_inst_ == nullptr && dbg_inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugInfoNone) { debug_info_none_inst_ = dbg_inst; } } void DebugInfoManager::AnalyzeDebugInsts(Module& module) { debug_info_none_inst_ = nullptr; module.ForEachInst([this](Instruction* cpi) { AnalyzeDebugInst(cpi); }); // Move |debug_info_none_inst_| to the beginning of the debug instruction // list. if (debug_info_none_inst_ != nullptr && debug_info_none_inst_->PreviousNode() != nullptr && debug_info_none_inst_->PreviousNode()->GetOpenCL100DebugOpcode() != OpenCLDebugInfo100InstructionsMax) { debug_info_none_inst_->InsertBefore( &*context()->module()->ext_inst_debuginfo_begin()); } } } // namespace analysis } // namespace opt } // namespace spvtools