| // Copyright (c) 2020-2022 Google LLC |
| // Copyright (c) 2022 LunarG 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/debug_info_manager.h" |
| |
| #include <cassert> |
| |
| #include "source/opt/ir_context.h" |
| |
| // Constants for OpenCL.DebugInfo.100 & NonSemantic.Shader.DebugInfo.100 |
| // extension instructions. |
| |
| namespace spvtools { |
| namespace opt { |
| namespace analysis { |
| namespace { |
| constexpr uint32_t kOpLineOperandLineIndex = 1; |
| constexpr uint32_t kLineOperandIndexDebugFunction = 7; |
| constexpr uint32_t kLineOperandIndexDebugLexicalBlock = 5; |
| constexpr uint32_t kLineOperandIndexDebugLine = 5; |
| constexpr uint32_t kDebugFunctionOperandFunctionIndex = 13; |
| constexpr uint32_t kDebugFunctionDefinitionOperandDebugFunctionIndex = 4; |
| constexpr uint32_t kDebugFunctionDefinitionOperandOpFunctionIndex = 5; |
| constexpr uint32_t kDebugFunctionOperandParentIndex = 9; |
| constexpr uint32_t kDebugTypeCompositeOperandParentIndex = 9; |
| constexpr uint32_t kDebugLexicalBlockOperandParentIndex = 7; |
| constexpr uint32_t kDebugInlinedAtOperandInlinedIndex = 6; |
| constexpr uint32_t kDebugExpressOperandOperationIndex = 4; |
| constexpr uint32_t kDebugDeclareOperandLocalVariableIndex = 4; |
| constexpr uint32_t kDebugDeclareOperandVariableIndex = 5; |
| constexpr uint32_t kDebugValueOperandExpressionIndex = 6; |
| constexpr uint32_t kDebugOperationOperandOperationIndex = 4; |
| constexpr uint32_t kOpVariableOperandStorageClassIndex = 2; |
| constexpr uint32_t kDebugLocalVariableOperandParentIndex = 9; |
| constexpr uint32_t kExtInstInstructionInIdx = 1; |
| constexpr uint32_t kDebugGlobalVariableOperandFlagsIndex = 12; |
| constexpr uint32_t kDebugLocalVariableOperandFlagsIndex = 10; |
| |
| void SetInlinedOperand(Instruction* dbg_inlined_at, uint32_t inlined_operand) { |
| assert(dbg_inlined_at); |
| assert(dbg_inlined_at->GetCommonDebugOpcode() == |
| CommonDebugInfoDebugInlinedAt); |
| if (dbg_inlined_at->NumOperands() <= kDebugInlinedAtOperandInlinedIndex) { |
| dbg_inlined_at->AddOperand( |
| {spv_operand_type_t::SPV_OPERAND_TYPE_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->GetCommonDebugOpcode() == |
| CommonDebugInfoDebugInlinedAt); |
| if (dbg_inlined_at->NumOperands() <= kDebugInlinedAtOperandInlinedIndex) |
| return kNoInlinedAt; |
| return dbg_inlined_at->GetSingleWordOperand( |
| kDebugInlinedAtOperandInlinedIndex); |
| } |
| |
| bool IsEmptyDebugExpression(Instruction* instr) { |
| return (instr->GetCommonDebugOpcode() == CommonDebugInfoDebugExpression) && |
| instr->NumOperands() == kDebugExpressOperandOperationIndex; |
| } |
| |
| } // namespace |
| |
| DebugInfoManager::DebugInfoManager(IRContext* c) : context_(c) { |
| AnalyzeDebugInsts(*c->module()); |
| } |
| |
| uint32_t DebugInfoManager::GetDbgSetImportId() { |
| uint32_t setId = |
| context()->get_feature_mgr()->GetExtInstImportId_OpenCL100DebugInfo(); |
| if (setId == 0) { |
| setId = |
| context()->get_feature_mgr()->GetExtInstImportId_Shader100DebugInfo(); |
| } |
| return setId; |
| } |
| |
| 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 && |
| (GetDbgSetImportId() == 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) { |
| if (inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugFunction) { |
| auto fn_id = inst->GetSingleWordOperand(kDebugFunctionOperandFunctionIndex); |
| // Do not register function that has been optimized away. |
| auto fn_inst = GetDbgInst(fn_id); |
| if (fn_inst != nullptr) { |
| assert(GetDbgInst(fn_id)->GetOpenCL100DebugOpcode() == |
| OpenCLDebugInfo100DebugInfoNone); |
| return; |
| } |
| 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; |
| } else if (inst->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugFunctionDefinition) { |
| auto fn_id = inst->GetSingleWordOperand( |
| kDebugFunctionDefinitionOperandOpFunctionIndex); |
| auto fn_inst = GetDbgInst(inst->GetSingleWordOperand( |
| kDebugFunctionDefinitionOperandDebugFunctionIndex)); |
| assert(fn_inst && fn_inst->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugFunction); |
| assert(fn_id_to_dbg_fn_.find(fn_id) == fn_id_to_dbg_fn_.end() && |
| "Register DebugFunctionDefinition for a function that already has " |
| "DebugFunctionDefinition"); |
| fn_id_to_dbg_fn_[fn_id] = fn_inst; |
| } else { |
| assert(false && "inst is not a DebugFunction"); |
| } |
| } |
| |
| void DebugInfoManager::RegisterDbgDeclare(uint32_t var_id, |
| Instruction* dbg_declare) { |
| assert(dbg_declare->GetCommonDebugOpcode() == CommonDebugInfoDebugDeclare || |
| dbg_declare->GetCommonDebugOpcode() == CommonDebugInfoDebugValue); |
| auto dbg_decl_itr = var_id_to_dbg_decl_.find(var_id); |
| if (dbg_decl_itr == var_id_to_dbg_decl_.end()) { |
| var_id_to_dbg_decl_[var_id] = {dbg_declare}; |
| } else { |
| dbg_decl_itr->second.insert(dbg_declare); |
| } |
| } |
| |
| // Create new constant directly into global value area, bypassing the |
| // Constant manager. This is used when the DefUse or Constant managers |
| // are invalid and cannot be regenerated due to the module being in an |
| // inconsistent state e.g. in the middle of significant modification |
| // such as inlining. Invalidate Constant and DefUse managers if used. |
| uint32_t AddNewConstInGlobals(IRContext* context, uint32_t const_value) { |
| uint32_t id = context->TakeNextId(); |
| std::unique_ptr<Instruction> new_const(new Instruction( |
| context, spv::Op::OpConstant, context->get_type_mgr()->GetUIntTypeId(), |
| id, |
| { |
| {spv_operand_type_t::SPV_OPERAND_TYPE_TYPED_LITERAL_NUMBER, |
| {const_value}}, |
| })); |
| context->module()->AddGlobalValue(std::move(new_const)); |
| context->InvalidateAnalyses(IRContext::kAnalysisConstants); |
| context->InvalidateAnalyses(IRContext::kAnalysisDefUse); |
| return id; |
| } |
| |
| uint32_t DebugInfoManager::CreateDebugInlinedAt(const Instruction* line, |
| const DebugScope& scope) { |
| uint32_t setId = GetDbgSetImportId(); |
| |
| if (setId == 0) return kNoInlinedAt; |
| |
| spv_operand_type_t line_number_type = |
| spv_operand_type_t::SPV_OPERAND_TYPE_LITERAL_INTEGER; |
| |
| // In NonSemantic.Shader.DebugInfo.100, all constants are IDs of OpConstant, |
| // not literals. |
| if (setId == |
| context()->get_feature_mgr()->GetExtInstImportId_Shader100DebugInfo()) |
| line_number_type = spv_operand_type_t::SPV_OPERAND_TYPE_ID; |
| |
| uint32_t line_number = 0; |
| if (line == nullptr) { |
| auto* lexical_scope_inst = GetDbgInst(scope.GetLexicalScope()); |
| if (lexical_scope_inst == nullptr) return kNoInlinedAt; |
| CommonDebugInfoInstructions debug_opcode = |
| lexical_scope_inst->GetCommonDebugOpcode(); |
| switch (debug_opcode) { |
| case CommonDebugInfoDebugFunction: |
| line_number = lexical_scope_inst->GetSingleWordOperand( |
| kLineOperandIndexDebugFunction); |
| break; |
| case CommonDebugInfoDebugLexicalBlock: |
| line_number = lexical_scope_inst->GetSingleWordOperand( |
| kLineOperandIndexDebugLexicalBlock); |
| break; |
| case CommonDebugInfoDebugTypeComposite: |
| case CommonDebugInfoDebugCompilationUnit: |
| 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 { |
| if (line->opcode() == spv::Op::OpLine) { |
| line_number = line->GetSingleWordOperand(kOpLineOperandLineIndex); |
| } else if (line->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugLine) { |
| line_number = line->GetSingleWordOperand(kLineOperandIndexDebugLine); |
| } else { |
| assert(false && |
| "Unreachable. A line instruction must be OpLine or DebugLine"); |
| } |
| |
| // If we need the line number as an ID, generate that constant now. |
| // If Constant or DefUse managers are invalid, generate constant |
| // directly into the global value section of the module; do not |
| // use Constant manager which may attempt to invoke building of the |
| // DefUse manager which cannot be done during inlining. The extra |
| // constants that may be generated here is likely not significant |
| // and will likely be cleaned up in later passes. |
| if (line_number_type == spv_operand_type_t::SPV_OPERAND_TYPE_ID && |
| line->opcode() == spv::Op::OpLine) { |
| if (!context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse) || |
| !context()->AreAnalysesValid(IRContext::Analysis::kAnalysisConstants)) |
| line_number = AddNewConstInGlobals(context(), line_number); |
| else |
| line_number = |
| context()->get_constant_mgr()->GetUIntConstId(line_number); |
| } |
| } |
| |
| uint32_t result_id = context()->TakeNextId(); |
| std::unique_ptr<Instruction> inlined_at(new Instruction( |
| context(), spv::Op::OpExtInst, context()->get_type_mgr()->GetVoidTypeId(), |
| result_id, |
| { |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {setId}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>(CommonDebugInfoDebugInlinedAt)}}, |
| {line_number_type, {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_ID, {scope.GetInlinedAt()}}); |
| } |
| RegisterDbgInst(inlined_at.get()); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(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::GetDebugOperationWithDeref() { |
| if (deref_operation_ != nullptr) return deref_operation_; |
| |
| uint32_t result_id = context()->TakeNextId(); |
| std::unique_ptr<Instruction> deref_operation; |
| |
| if (context()->get_feature_mgr()->GetExtInstImportId_OpenCL100DebugInfo()) { |
| deref_operation = std::unique_ptr<Instruction>(new Instruction( |
| context(), spv::Op::OpExtInst, |
| context()->get_type_mgr()->GetVoidTypeId(), result_id, |
| { |
| {SPV_OPERAND_TYPE_ID, {GetDbgSetImportId()}}, |
| {SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>(OpenCLDebugInfo100DebugOperation)}}, |
| {SPV_OPERAND_TYPE_CLDEBUG100_DEBUG_OPERATION, |
| {static_cast<uint32_t>(OpenCLDebugInfo100Deref)}}, |
| })); |
| } else { |
| uint32_t deref_id = context()->get_constant_mgr()->GetUIntConstId( |
| NonSemanticShaderDebugInfo100Deref); |
| |
| deref_operation = std::unique_ptr<Instruction>( |
| new Instruction(context(), spv::Op::OpExtInst, |
| context()->get_type_mgr()->GetVoidTypeId(), result_id, |
| { |
| {SPV_OPERAND_TYPE_ID, {GetDbgSetImportId()}}, |
| {SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>( |
| NonSemanticShaderDebugInfo100DebugOperation)}}, |
| {SPV_OPERAND_TYPE_ID, {deref_id}}, |
| })); |
| } |
| |
| // Add to the front of |ext_inst_debuginfo_|. |
| deref_operation_ = |
| context()->module()->ext_inst_debuginfo_begin()->InsertBefore( |
| std::move(deref_operation)); |
| |
| RegisterDbgInst(deref_operation_); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(deref_operation_); |
| return deref_operation_; |
| } |
| |
| Instruction* DebugInfoManager::DerefDebugExpression(Instruction* dbg_expr) { |
| assert(dbg_expr->GetCommonDebugOpcode() == CommonDebugInfoDebugExpression); |
| std::unique_ptr<Instruction> deref_expr(dbg_expr->Clone(context())); |
| deref_expr->SetResultId(context()->TakeNextId()); |
| deref_expr->InsertOperand( |
| kDebugExpressOperandOperationIndex, |
| {SPV_OPERAND_TYPE_ID, {GetDebugOperationWithDeref()->result_id()}}); |
| auto* deref_expr_instr = |
| context()->ext_inst_debuginfo_end()->InsertBefore(std::move(deref_expr)); |
| AnalyzeDebugInst(deref_expr_instr); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(deref_expr_instr); |
| return deref_expr_instr; |
| } |
| |
| Instruction* DebugInfoManager::GetDebugInfoNone() { |
| if (debug_info_none_inst_ != nullptr) return debug_info_none_inst_; |
| |
| uint32_t result_id = context()->TakeNextId(); |
| std::unique_ptr<Instruction> dbg_info_none_inst(new Instruction( |
| context(), spv::Op::OpExtInst, context()->get_type_mgr()->GetVoidTypeId(), |
| result_id, |
| { |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {GetDbgSetImportId()}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>(CommonDebugInfoDebugInfoNone)}}, |
| })); |
| |
| // 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_); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(debug_info_none_inst_); |
| return debug_info_none_inst_; |
| } |
| |
| Instruction* DebugInfoManager::GetEmptyDebugExpression() { |
| if (empty_debug_expr_inst_ != nullptr) return empty_debug_expr_inst_; |
| |
| uint32_t result_id = context()->TakeNextId(); |
| std::unique_ptr<Instruction> empty_debug_expr(new Instruction( |
| context(), spv::Op::OpExtInst, context()->get_type_mgr()->GetVoidTypeId(), |
| result_id, |
| { |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {GetDbgSetImportId()}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>(CommonDebugInfoDebugExpression)}}, |
| })); |
| |
| // Add to the front of |ext_inst_debuginfo_|. |
| empty_debug_expr_inst_ = |
| context()->module()->ext_inst_debuginfo_begin()->InsertBefore( |
| std::move(empty_debug_expr)); |
| |
| RegisterDbgInst(empty_debug_expr_inst_); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(empty_debug_expr_inst_); |
| return empty_debug_expr_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->GetCommonDebugOpcode() != CommonDebugInfoDebugInlinedAt) { |
| 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<Instruction> new_inlined_at(inlined_at->Clone(context())); |
| new_inlined_at->SetResultId(context()->TakeNextId()); |
| RegisterDbgInst(new_inlined_at.get()); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(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)); |
| } |
| |
| bool DebugInfoManager::IsVariableDebugDeclared(uint32_t variable_id) { |
| auto dbg_decl_itr = var_id_to_dbg_decl_.find(variable_id); |
| return dbg_decl_itr != var_id_to_dbg_decl_.end(); |
| } |
| |
| bool DebugInfoManager::KillDebugDeclares(uint32_t variable_id) { |
| bool modified = false; |
| auto dbg_decl_itr = var_id_to_dbg_decl_.find(variable_id); |
| if (dbg_decl_itr != var_id_to_dbg_decl_.end()) { |
| // We intentionally copy the list of DebugDeclare instructions because |
| // context()->KillInst(dbg_decl) will update |var_id_to_dbg_decl_|. If we |
| // directly use |dbg_decl_itr->second|, it accesses a dangling pointer. |
| auto copy_dbg_decls = dbg_decl_itr->second; |
| |
| for (auto* dbg_decl : copy_dbg_decls) { |
| context()->KillInst(dbg_decl); |
| modified = true; |
| } |
| var_id_to_dbg_decl_.erase(dbg_decl_itr); |
| } |
| return modified; |
| } |
| |
| uint32_t DebugInfoManager::GetParentScope(uint32_t child_scope) { |
| auto dbg_scope_itr = id_to_dbg_inst_.find(child_scope); |
| assert(dbg_scope_itr != id_to_dbg_inst_.end()); |
| CommonDebugInfoInstructions debug_opcode = |
| dbg_scope_itr->second->GetCommonDebugOpcode(); |
| uint32_t parent_scope = kNoDebugScope; |
| switch (debug_opcode) { |
| case CommonDebugInfoDebugFunction: |
| parent_scope = dbg_scope_itr->second->GetSingleWordOperand( |
| kDebugFunctionOperandParentIndex); |
| break; |
| case CommonDebugInfoDebugLexicalBlock: |
| parent_scope = dbg_scope_itr->second->GetSingleWordOperand( |
| kDebugLexicalBlockOperandParentIndex); |
| break; |
| case CommonDebugInfoDebugTypeComposite: |
| parent_scope = dbg_scope_itr->second->GetSingleWordOperand( |
| kDebugTypeCompositeOperandParentIndex); |
| break; |
| case CommonDebugInfoDebugCompilationUnit: |
| // DebugCompilationUnit does not have a parent scope. |
| break; |
| default: |
| assert(false && |
| "Unreachable. A debug scope instruction must be " |
| "DebugFunction, DebugTypeComposite, DebugLexicalBlock, " |
| "or DebugCompilationUnit."); |
| break; |
| } |
| return parent_scope; |
| } |
| |
| bool DebugInfoManager::IsAncestorOfScope(uint32_t scope, uint32_t ancestor) { |
| uint32_t ancestor_scope_itr = scope; |
| while (ancestor_scope_itr != kNoDebugScope) { |
| if (ancestor == ancestor_scope_itr) return true; |
| ancestor_scope_itr = GetParentScope(ancestor_scope_itr); |
| } |
| return false; |
| } |
| |
| bool DebugInfoManager::IsDeclareVisibleToInstr(Instruction* dbg_declare, |
| Instruction* scope) { |
| assert(dbg_declare != nullptr); |
| assert(scope != nullptr); |
| |
| std::vector<uint32_t> scope_ids; |
| if (scope->opcode() == spv::Op::OpPhi) { |
| scope_ids.push_back(scope->GetDebugScope().GetLexicalScope()); |
| for (uint32_t i = 0; i < scope->NumInOperands(); i += 2) { |
| auto* value = context()->get_def_use_mgr()->GetDef( |
| scope->GetSingleWordInOperand(i)); |
| if (value != nullptr) |
| scope_ids.push_back(value->GetDebugScope().GetLexicalScope()); |
| } |
| } else { |
| scope_ids.push_back(scope->GetDebugScope().GetLexicalScope()); |
| } |
| |
| uint32_t dbg_local_var_id = |
| dbg_declare->GetSingleWordOperand(kDebugDeclareOperandLocalVariableIndex); |
| auto dbg_local_var_itr = id_to_dbg_inst_.find(dbg_local_var_id); |
| assert(dbg_local_var_itr != id_to_dbg_inst_.end()); |
| uint32_t decl_scope_id = dbg_local_var_itr->second->GetSingleWordOperand( |
| kDebugLocalVariableOperandParentIndex); |
| |
| // If the scope of DebugDeclare is an ancestor scope of the instruction's |
| // scope, the local variable is visible to the instruction. |
| for (uint32_t scope_id : scope_ids) { |
| if (scope_id != kNoDebugScope && |
| IsAncestorOfScope(scope_id, decl_scope_id)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| bool DebugInfoManager::AddDebugValueForVariable(Instruction* scope_and_line, |
| uint32_t variable_id, |
| uint32_t value_id, |
| Instruction* insert_pos) { |
| assert(scope_and_line != nullptr); |
| |
| auto dbg_decl_itr = var_id_to_dbg_decl_.find(variable_id); |
| if (dbg_decl_itr == var_id_to_dbg_decl_.end()) return false; |
| |
| bool modified = false; |
| for (auto* dbg_decl_or_val : dbg_decl_itr->second) { |
| // Avoid inserting the new DebugValue between OpPhi or OpVariable |
| // instructions. |
| Instruction* insert_before = insert_pos->NextNode(); |
| while (insert_before->opcode() == spv::Op::OpPhi || |
| insert_before->opcode() == spv::Op::OpVariable) { |
| insert_before = insert_before->NextNode(); |
| } |
| modified |= AddDebugValueForDecl(dbg_decl_or_val, value_id, insert_before, |
| scope_and_line) != nullptr; |
| } |
| return modified; |
| } |
| |
| Instruction* DebugInfoManager::AddDebugValueForDecl( |
| Instruction* dbg_decl, uint32_t value_id, Instruction* insert_before, |
| Instruction* scope_and_line) { |
| if (dbg_decl == nullptr || !IsDebugDeclare(dbg_decl)) return nullptr; |
| |
| std::unique_ptr<Instruction> dbg_val(dbg_decl->Clone(context())); |
| dbg_val->SetResultId(context()->TakeNextId()); |
| dbg_val->SetInOperand(kExtInstInstructionInIdx, {CommonDebugInfoDebugValue}); |
| dbg_val->SetOperand(kDebugDeclareOperandVariableIndex, {value_id}); |
| dbg_val->SetOperand(kDebugValueOperandExpressionIndex, |
| {GetEmptyDebugExpression()->result_id()}); |
| dbg_val->UpdateDebugInfoFrom(scope_and_line); |
| |
| auto* added_dbg_val = insert_before->InsertBefore(std::move(dbg_val)); |
| AnalyzeDebugInst(added_dbg_val); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(added_dbg_val); |
| if (context()->AreAnalysesValid( |
| IRContext::Analysis::kAnalysisInstrToBlockMapping)) { |
| auto insert_blk = context()->get_instr_block(insert_before); |
| context()->set_instr_block(added_dbg_val, insert_blk); |
| } |
| return added_dbg_val; |
| } |
| |
| uint32_t DebugInfoManager::GetVulkanDebugOperation(Instruction* inst) { |
| assert(inst->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugOperation && |
| "inst must be Vulkan DebugOperation"); |
| return context() |
| ->get_constant_mgr() |
| ->GetConstantFromInst(context()->get_def_use_mgr()->GetDef( |
| inst->GetSingleWordOperand(kDebugOperationOperandOperationIndex))) |
| ->GetU32(); |
| } |
| |
| uint32_t DebugInfoManager::GetVariableIdOfDebugValueUsedForDeclare( |
| Instruction* inst) { |
| if (inst->GetCommonDebugOpcode() != CommonDebugInfoDebugValue) return 0; |
| |
| auto* expr = |
| GetDbgInst(inst->GetSingleWordOperand(kDebugValueOperandExpressionIndex)); |
| if (expr == nullptr) return 0; |
| if (expr->NumOperands() != kDebugExpressOperandOperationIndex + 1) return 0; |
| |
| auto* operation = GetDbgInst( |
| expr->GetSingleWordOperand(kDebugExpressOperandOperationIndex)); |
| if (operation == nullptr) return 0; |
| |
| // OpenCL.DebugInfo.100 contains a literal for the operation, Vulkan uses an |
| // OpConstant. |
| if (inst->IsOpenCL100DebugInstr()) { |
| if (operation->GetSingleWordOperand(kDebugOperationOperandOperationIndex) != |
| OpenCLDebugInfo100Deref) { |
| return 0; |
| } |
| } else { |
| uint32_t operation_const = GetVulkanDebugOperation(operation); |
| if (operation_const != NonSemanticShaderDebugInfo100Deref) { |
| return 0; |
| } |
| } |
| |
| uint32_t var_id = |
| inst->GetSingleWordOperand(kDebugDeclareOperandVariableIndex); |
| if (!context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) { |
| assert(false && |
| "Checking a DebugValue can be used for declare needs DefUseManager"); |
| return 0; |
| } |
| |
| auto* var = context()->get_def_use_mgr()->GetDef(var_id); |
| if (var->opcode() == spv::Op::OpVariable && |
| spv::StorageClass( |
| var->GetSingleWordOperand(kOpVariableOperandStorageClassIndex)) == |
| spv::StorageClass::Function) { |
| return var_id; |
| } |
| return 0; |
| } |
| |
| bool DebugInfoManager::IsDebugDeclare(Instruction* instr) { |
| if (!instr->IsCommonDebugInstr()) return false; |
| return instr->GetCommonDebugOpcode() == CommonDebugInfoDebugDeclare || |
| GetVariableIdOfDebugValueUsedForDeclare(instr) != 0; |
| } |
| |
| void DebugInfoManager::ReplaceAllUsesInDebugScopeWithPredicate( |
| uint32_t before, uint32_t after, |
| const std::function<bool(Instruction*)>& predicate) { |
| auto scope_id_to_users_itr = scope_id_to_users_.find(before); |
| if (scope_id_to_users_itr != scope_id_to_users_.end()) { |
| for (Instruction* inst : scope_id_to_users_itr->second) { |
| if (predicate(inst)) inst->UpdateLexicalScope(after); |
| } |
| scope_id_to_users_[after] = scope_id_to_users_itr->second; |
| scope_id_to_users_.erase(scope_id_to_users_itr); |
| } |
| auto inlinedat_id_to_users_itr = inlinedat_id_to_users_.find(before); |
| if (inlinedat_id_to_users_itr != inlinedat_id_to_users_.end()) { |
| for (Instruction* inst : inlinedat_id_to_users_itr->second) { |
| if (predicate(inst)) inst->UpdateDebugInlinedAt(after); |
| } |
| inlinedat_id_to_users_[after] = inlinedat_id_to_users_itr->second; |
| inlinedat_id_to_users_.erase(inlinedat_id_to_users_itr); |
| } |
| } |
| |
| void DebugInfoManager::ClearDebugScopeAndInlinedAtUses(Instruction* inst) { |
| auto scope_id_to_users_itr = scope_id_to_users_.find(inst->result_id()); |
| if (scope_id_to_users_itr != scope_id_to_users_.end()) { |
| scope_id_to_users_.erase(scope_id_to_users_itr); |
| } |
| auto inlinedat_id_to_users_itr = |
| inlinedat_id_to_users_.find(inst->result_id()); |
| if (inlinedat_id_to_users_itr != inlinedat_id_to_users_.end()) { |
| inlinedat_id_to_users_.erase(inlinedat_id_to_users_itr); |
| } |
| } |
| |
| void DebugInfoManager::AnalyzeDebugInst(Instruction* inst) { |
| if (inst->GetDebugScope().GetLexicalScope() != kNoDebugScope) { |
| auto& users = scope_id_to_users_[inst->GetDebugScope().GetLexicalScope()]; |
| users.insert(inst); |
| } |
| if (inst->GetDebugInlinedAt() != kNoInlinedAt) { |
| auto& users = inlinedat_id_to_users_[inst->GetDebugInlinedAt()]; |
| users.insert(inst); |
| } |
| |
| if (!inst->IsCommonDebugInstr()) return; |
| |
| RegisterDbgInst(inst); |
| |
| if (inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugFunction || |
| inst->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugFunctionDefinition) { |
| RegisterDbgFunction(inst); |
| } |
| |
| if (deref_operation_ == nullptr && |
| inst->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugOperation && |
| inst->GetSingleWordOperand(kDebugOperationOperandOperationIndex) == |
| OpenCLDebugInfo100Deref) { |
| deref_operation_ = inst; |
| } |
| |
| if (deref_operation_ == nullptr && |
| inst->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugOperation) { |
| uint32_t operation_const = GetVulkanDebugOperation(inst); |
| if (operation_const == NonSemanticShaderDebugInfo100Deref) { |
| deref_operation_ = inst; |
| } |
| } |
| |
| if (debug_info_none_inst_ == nullptr && |
| inst->GetCommonDebugOpcode() == CommonDebugInfoDebugInfoNone) { |
| debug_info_none_inst_ = inst; |
| } |
| |
| if (empty_debug_expr_inst_ == nullptr && IsEmptyDebugExpression(inst)) { |
| empty_debug_expr_inst_ = inst; |
| } |
| |
| if (inst->GetCommonDebugOpcode() == CommonDebugInfoDebugDeclare) { |
| uint32_t var_id = |
| inst->GetSingleWordOperand(kDebugDeclareOperandVariableIndex); |
| RegisterDbgDeclare(var_id, inst); |
| } |
| |
| if (uint32_t var_id = GetVariableIdOfDebugValueUsedForDeclare(inst)) { |
| RegisterDbgDeclare(var_id, inst); |
| } |
| } |
| |
| void DebugInfoManager::ConvertDebugGlobalToLocalVariable( |
| Instruction* dbg_global_var, Instruction* local_var) { |
| if (dbg_global_var->GetCommonDebugOpcode() != |
| CommonDebugInfoDebugGlobalVariable) { |
| return; |
| } |
| assert(local_var->opcode() == spv::Op::OpVariable || |
| local_var->opcode() == spv::Op::OpFunctionParameter); |
| |
| // Convert |dbg_global_var| to DebugLocalVariable |
| dbg_global_var->SetInOperand(kExtInstInstructionInIdx, |
| {CommonDebugInfoDebugLocalVariable}); |
| auto flags = dbg_global_var->GetSingleWordOperand( |
| kDebugGlobalVariableOperandFlagsIndex); |
| for (uint32_t i = dbg_global_var->NumInOperands() - 1; |
| i >= kDebugLocalVariableOperandFlagsIndex; --i) { |
| dbg_global_var->RemoveOperand(i); |
| } |
| dbg_global_var->SetOperand(kDebugLocalVariableOperandFlagsIndex, {flags}); |
| context()->ForgetUses(dbg_global_var); |
| context()->AnalyzeUses(dbg_global_var); |
| |
| // Create a DebugDeclare |
| std::unique_ptr<Instruction> new_dbg_decl(new Instruction( |
| context(), spv::Op::OpExtInst, context()->get_type_mgr()->GetVoidTypeId(), |
| context()->TakeNextId(), |
| { |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {GetDbgSetImportId()}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_EXTENSION_INSTRUCTION_NUMBER, |
| {static_cast<uint32_t>(CommonDebugInfoDebugDeclare)}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, |
| {dbg_global_var->result_id()}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, {local_var->result_id()}}, |
| {spv_operand_type_t::SPV_OPERAND_TYPE_ID, |
| {GetEmptyDebugExpression()->result_id()}}, |
| })); |
| // Must insert after all OpVariables in block |
| Instruction* insert_before = local_var; |
| while (insert_before->opcode() == spv::Op::OpVariable) |
| insert_before = insert_before->NextNode(); |
| auto* added_dbg_decl = insert_before->InsertBefore(std::move(new_dbg_decl)); |
| if (context()->AreAnalysesValid(IRContext::Analysis::kAnalysisDefUse)) |
| context()->get_def_use_mgr()->AnalyzeInstDefUse(added_dbg_decl); |
| if (context()->AreAnalysesValid( |
| IRContext::Analysis::kAnalysisInstrToBlockMapping)) { |
| auto insert_blk = context()->get_instr_block(local_var); |
| context()->set_instr_block(added_dbg_decl, insert_blk); |
| } |
| } |
| |
| void DebugInfoManager::AnalyzeDebugInsts(Module& module) { |
| deref_operation_ = nullptr; |
| debug_info_none_inst_ = nullptr; |
| empty_debug_expr_inst_ = nullptr; |
| module.ForEachInst([this](Instruction* cpi) { AnalyzeDebugInst(cpi); }); |
| |
| // Move |empty_debug_expr_inst_| to the beginning of the debug instruction |
| // list. |
| if (empty_debug_expr_inst_ != nullptr && |
| empty_debug_expr_inst_->PreviousNode() != nullptr && |
| empty_debug_expr_inst_->PreviousNode()->IsCommonDebugInstr()) { |
| empty_debug_expr_inst_->InsertBefore( |
| &*context()->module()->ext_inst_debuginfo_begin()); |
| } |
| |
| // 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()->IsCommonDebugInstr()) { |
| debug_info_none_inst_->InsertBefore( |
| &*context()->module()->ext_inst_debuginfo_begin()); |
| } |
| } |
| |
| void DebugInfoManager::ClearDebugInfo(Instruction* instr) { |
| auto scope_id_to_users_itr = |
| scope_id_to_users_.find(instr->GetDebugScope().GetLexicalScope()); |
| if (scope_id_to_users_itr != scope_id_to_users_.end()) { |
| scope_id_to_users_itr->second.erase(instr); |
| } |
| auto inlinedat_id_to_users_itr = |
| inlinedat_id_to_users_.find(instr->GetDebugInlinedAt()); |
| if (inlinedat_id_to_users_itr != inlinedat_id_to_users_.end()) { |
| inlinedat_id_to_users_itr->second.erase(instr); |
| } |
| |
| if (instr == nullptr || !instr->IsCommonDebugInstr()) { |
| return; |
| } |
| |
| id_to_dbg_inst_.erase(instr->result_id()); |
| |
| if (instr->GetOpenCL100DebugOpcode() == OpenCLDebugInfo100DebugFunction) { |
| auto fn_id = |
| instr->GetSingleWordOperand(kDebugFunctionOperandFunctionIndex); |
| fn_id_to_dbg_fn_.erase(fn_id); |
| } |
| if (instr->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugFunctionDefinition) { |
| auto fn_id = instr->GetSingleWordOperand( |
| kDebugFunctionDefinitionOperandOpFunctionIndex); |
| fn_id_to_dbg_fn_.erase(fn_id); |
| } |
| |
| if (instr->GetCommonDebugOpcode() == CommonDebugInfoDebugDeclare || |
| instr->GetCommonDebugOpcode() == CommonDebugInfoDebugValue) { |
| auto var_or_value_id = |
| instr->GetSingleWordOperand(kDebugDeclareOperandVariableIndex); |
| auto dbg_decl_itr = var_id_to_dbg_decl_.find(var_or_value_id); |
| if (dbg_decl_itr != var_id_to_dbg_decl_.end()) { |
| dbg_decl_itr->second.erase(instr); |
| } |
| } |
| |
| if (deref_operation_ == instr) { |
| deref_operation_ = nullptr; |
| for (auto dbg_instr_itr = context()->module()->ext_inst_debuginfo_begin(); |
| dbg_instr_itr != context()->module()->ext_inst_debuginfo_end(); |
| ++dbg_instr_itr) { |
| // OpenCL.DebugInfo.100 contains the operation as a literal operand, in |
| // Vulkan it's referenced as an OpConstant. |
| if (instr != &*dbg_instr_itr && |
| dbg_instr_itr->GetOpenCL100DebugOpcode() == |
| OpenCLDebugInfo100DebugOperation && |
| dbg_instr_itr->GetSingleWordOperand( |
| kDebugOperationOperandOperationIndex) == |
| OpenCLDebugInfo100Deref) { |
| deref_operation_ = &*dbg_instr_itr; |
| break; |
| } else if (instr != &*dbg_instr_itr && |
| dbg_instr_itr->GetShader100DebugOpcode() == |
| NonSemanticShaderDebugInfo100DebugOperation) { |
| uint32_t operation_const = GetVulkanDebugOperation(&*dbg_instr_itr); |
| if (operation_const == NonSemanticShaderDebugInfo100Deref) { |
| deref_operation_ = &*dbg_instr_itr; |
| break; |
| } |
| } |
| } |
| } |
| |
| if (debug_info_none_inst_ == instr) { |
| debug_info_none_inst_ = nullptr; |
| for (auto dbg_instr_itr = context()->module()->ext_inst_debuginfo_begin(); |
| dbg_instr_itr != context()->module()->ext_inst_debuginfo_end(); |
| ++dbg_instr_itr) { |
| if (instr != &*dbg_instr_itr && dbg_instr_itr->GetCommonDebugOpcode() == |
| CommonDebugInfoDebugInfoNone) { |
| debug_info_none_inst_ = &*dbg_instr_itr; |
| break; |
| } |
| } |
| } |
| |
| if (empty_debug_expr_inst_ == instr) { |
| empty_debug_expr_inst_ = nullptr; |
| for (auto dbg_instr_itr = context()->module()->ext_inst_debuginfo_begin(); |
| dbg_instr_itr != context()->module()->ext_inst_debuginfo_end(); |
| ++dbg_instr_itr) { |
| if (instr != &*dbg_instr_itr && IsEmptyDebugExpression(&*dbg_instr_itr)) { |
| empty_debug_expr_inst_ = &*dbg_instr_itr; |
| break; |
| } |
| } |
| } |
| } |
| |
| } // namespace analysis |
| } // namespace opt |
| } // namespace spvtools |