diagnostic_helpers.h 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  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. #ifndef CARBON_TOOLCHAIN_CHECK_DIAGNOSTIC_HELPERS_H_
  5. #define CARBON_TOOLCHAIN_CHECK_DIAGNOSTIC_HELPERS_H_
  6. #include "llvm/ADT/APSInt.h"
  7. #include "toolchain/parse/node_ids.h"
  8. #include "toolchain/sem_ir/ids.h"
  9. namespace Carbon::Check {
  10. // TODO: Consider instead changing calls to `SemIR::LocId::TokenOnly(...)`.
  11. inline auto TokenOnly(SemIR::LocId loc_id) -> SemIR::LocId {
  12. return loc_id.ToTokenOnly();
  13. }
  14. // We define the emitter separately for dependencies, so only provide a base
  15. // here.
  16. using DiagnosticEmitterBase = Diagnostics::Emitter<SemIR::LocId>;
  17. using DiagnosticBuilder = DiagnosticEmitterBase::Builder;
  18. // A function that forms a diagnostic for some kind of problem. The
  19. // DiagnosticBuilder is returned rather than emitted so that the caller
  20. // can add contextual notes as appropriate.
  21. using MakeDiagnosticBuilderFn = llvm::function_ref<auto()->DiagnosticBuilder>;
  22. // An expression with a constant value, for rendering in a diagnostic. The
  23. // diagnostic rendering will include enclosing "`"s.
  24. struct InstIdAsConstant {
  25. using DiagnosticType = Diagnostics::TypeInfo<std::string>;
  26. // NOLINTNEXTLINE(google-explicit-constructor)
  27. InstIdAsConstant(SemIR::InstId inst_id) : inst_id(inst_id) {}
  28. SemIR::InstId inst_id;
  29. };
  30. // An expression whose type should be rendered in a diagnostic. The diagnostic
  31. // rendering will include enclosing "`"s, and may also include extra information
  32. // about the type if it might otherwise be ambiguous or context-dependent, such
  33. // as the targets of aliases used in the type.
  34. //
  35. // TODO: Include such additional information where relevant. For example:
  36. // "`StdString` (aka `Cpp.std.basic_string(Char)`)".
  37. //
  38. // This should be used instead of `TypeId` as a diagnostic argument wherever
  39. // possible, because we should eventually be able to produce a sugared type name
  40. // in this case, whereas a `TypeId` will render as a canonical type.
  41. struct TypeOfInstId {
  42. using DiagnosticType = Diagnostics::TypeInfo<std::string>;
  43. // NOLINTNEXTLINE(google-explicit-constructor)
  44. TypeOfInstId(SemIR::InstId inst_id) : inst_id(inst_id) {}
  45. SemIR::InstId inst_id;
  46. };
  47. // A type expression, for rendering in a diagnostic. The diagnostic rendering
  48. // will include enclosing "`"s, and may also include extra information about the
  49. // type if it would otherwise be ambiguous.
  50. //
  51. // TODO: Include such additional information where relevant.
  52. //
  53. // This should be used when the source expression used to construct a type is
  54. // available.
  55. //
  56. // Note that this is currently an alias for InstIdAsConstant. However, using
  57. // InstIdAsType is clearer when defining CARBON_DIAGNOSTICs, and we may wish to
  58. // distinguish type arguments in diagnostics from more general constants in some
  59. // way in the future.
  60. using InstIdAsType = InstIdAsConstant;
  61. // A type expression, for rendering in a diagnostic as a raw type. When
  62. // formatting as a raw type in a diagnostic, the type will be formatted as a
  63. // simple Carbon expression, without enclosing "`"s. Once we start including
  64. // extra information about types, such annotations will also not be included for
  65. // raw types.
  66. //
  67. // This is intended for cases where the type is part of a larger syntactic
  68. // construct in a diagnostic, such as "redefinition of `impl {0} as {1}`".
  69. struct InstIdAsRawType {
  70. using DiagnosticType = Diagnostics::TypeInfo<std::string>;
  71. // NOLINTNEXTLINE(google-explicit-constructor)
  72. InstIdAsRawType(SemIR::InstId inst_id) : inst_id(inst_id) {}
  73. SemIR::InstId inst_id;
  74. };
  75. // A type value for rendering in a diagnostic without enclosing "`"s. See
  76. // `InstIdAsRawType` for details on raw type formatting.
  77. //
  78. // As with `TypeId`, this should be avoided as a diagnostic argument where
  79. // possible, because it can't be formatted with syntactic sugar such as aliases
  80. // that describe how the type was written.
  81. struct TypeIdAsRawType {
  82. using DiagnosticType = Diagnostics::TypeInfo<std::string>;
  83. // NOLINTNEXTLINE(google-explicit-constructor)
  84. TypeIdAsRawType(SemIR::TypeId type_id) : type_id(type_id) {}
  85. SemIR::TypeId type_id;
  86. };
  87. // An integer value together with its type. The type is used to determine how to
  88. // format the value in diagnostics.
  89. struct TypedInt {
  90. using DiagnosticType = Diagnostics::TypeInfo<llvm::APSInt>;
  91. SemIR::TypeId type;
  92. llvm::APInt value;
  93. };
  94. } // namespace Carbon::Check
  95. #endif // CARBON_TOOLCHAIN_CHECK_DIAGNOSTIC_HELPERS_H_