context.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "toolchain/lower/context.h"
  5. #include "common/check.h"
  6. #include "common/vlog.h"
  7. #include "llvm/Transforms/Utils/ModuleUtils.h"
  8. #include "toolchain/lower/file_context.h"
  9. #include "toolchain/sem_ir/inst_namer.h"
  10. namespace Carbon::Lower {
  11. Context::Context(llvm::LLVMContext& llvm_context,
  12. llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> fs,
  13. std::optional<llvm::ArrayRef<Parse::GetTreeAndSubtreesFn>>
  14. tree_and_subtrees_getters_for_debug_info,
  15. llvm::StringRef module_name, llvm::raw_ostream* vlog_stream)
  16. : llvm_context_(&llvm_context),
  17. llvm_module_(std::make_unique<llvm::Module>(module_name, llvm_context)),
  18. file_system_(std::move(fs)),
  19. di_builder_(*llvm_module_),
  20. di_compile_unit_(
  21. tree_and_subtrees_getters_for_debug_info
  22. ? BuildDICompileUnit(module_name, *llvm_module_, di_builder_)
  23. : nullptr),
  24. tree_and_subtrees_getters_for_debug_info_(
  25. tree_and_subtrees_getters_for_debug_info),
  26. vlog_stream_(vlog_stream) {}
  27. auto Context::GetFileContext(const SemIR::File* file,
  28. const SemIR::InstNamer* inst_namer)
  29. -> FileContext& {
  30. auto insert_result = file_contexts_.Insert(file->check_ir_id(), [&] {
  31. auto file_context =
  32. std::make_unique<FileContext>(*this, *file, inst_namer, vlog_stream_);
  33. file_context->PrepareToLower();
  34. return file_context;
  35. });
  36. return *insert_result.value();
  37. }
  38. auto Context::LowerPendingDefinitions() -> void {
  39. // Lower function definitions for generics.
  40. // This cannot be a range-based loop, as new definitions can be added
  41. // while building other definitions.
  42. // NOLINTNEXTLINE(modernize-loop-convert)
  43. for (size_t i = 0; i != specific_function_definitions_.size(); ++i) {
  44. auto [file_context, function_id, specific_id] =
  45. specific_function_definitions_[i];
  46. file_context->BuildFunctionDefinition(function_id, specific_id);
  47. }
  48. }
  49. auto Context::Finalize() && -> std::unique_ptr<llvm::Module> {
  50. LowerPendingDefinitions();
  51. file_contexts_.ForEach(
  52. [](auto, auto& file_context) { file_context->Finalize(); });
  53. return std::move(llvm_module_);
  54. }
  55. auto Context::BuildDICompileUnit(llvm::StringRef module_name,
  56. llvm::Module& llvm_module,
  57. llvm::DIBuilder& di_builder)
  58. -> llvm::DICompileUnit* {
  59. llvm_module.addModuleFlag(llvm::Module::Max, "Dwarf Version", 5);
  60. llvm_module.addModuleFlag(llvm::Module::Warning, "Debug Info Version",
  61. llvm::DEBUG_METADATA_VERSION);
  62. // TODO: Include directory path in the compile_unit_file.
  63. llvm::DIFile* compile_unit_file = di_builder.createFile(module_name, "");
  64. // TODO: Introduce a new language code for Carbon. C works well for now since
  65. // it's something debuggers will already know/have support for at least.
  66. // Probably have to bump to C++ at some point for virtual functions,
  67. // templates, etc.
  68. return di_builder.createCompileUnit(llvm::dwarf::DW_LANG_C, compile_unit_file,
  69. "carbon",
  70. /*isOptimized=*/false, /*Flags=*/"",
  71. /*RV=*/0);
  72. }
  73. auto Context::GetLocForDI(SemIR::AbsoluteNodeId abs_node_id) -> LocForDI {
  74. const auto& tree_and_subtrees =
  75. (*tree_and_subtrees_getters_for_debug_info_)[abs_node_id.check_ir_id()
  76. .index]();
  77. const auto& tokens = tree_and_subtrees.tree().tokens();
  78. if (abs_node_id.node_id().has_value()) {
  79. auto token =
  80. tree_and_subtrees.GetSubtreeTokenRange(abs_node_id.node_id()).begin;
  81. return {.filename = tokens.source().filename(),
  82. .line_number = tokens.GetLineNumber(token),
  83. .column_number = tokens.GetColumnNumber(token)};
  84. } else {
  85. return {.filename = tokens.source().filename(),
  86. .line_number = 0,
  87. .column_number = 0};
  88. }
  89. }
  90. } // namespace Carbon::Lower