ids_test.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  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/sem_ir/ids.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include <limits>
  8. #include <tuple>
  9. namespace Carbon::SemIR {
  10. namespace {
  11. using ::testing::Eq;
  12. TEST(IdsTest, LocIdValues) {
  13. // This testing should match the ranges documented on LocId.
  14. EXPECT_THAT(static_cast<LocId>(Parse::NodeId::None).index, Eq(-1));
  15. EXPECT_THAT(static_cast<LocId>(InstId(0)).index, Eq(0));
  16. EXPECT_THAT(
  17. static_cast<LocId>(InstId(std::numeric_limits<int32_t>::max())).index,
  18. Eq(std::numeric_limits<int32_t>::max()));
  19. auto min_node_id = static_cast<LocId>(Parse::NodeId(0));
  20. EXPECT_THAT(min_node_id.index, Eq(-2));
  21. EXPECT_THAT(min_node_id.AsDesugared().index, Eq(-2 - (1 << 24)));
  22. EXPECT_THAT(min_node_id.AsDesugared().index,
  23. Eq(min_node_id.AsDesugared().AsDesugared().index));
  24. auto max_node_id = static_cast<LocId>(Parse::NodeId(Parse::NodeId::Max - 1));
  25. EXPECT_THAT(max_node_id.index, Eq(-2 - (1 << 24) + 1));
  26. EXPECT_THAT(max_node_id.AsDesugared().index, Eq(-2 - (1 << 25) + 1));
  27. EXPECT_THAT(max_node_id.AsDesugared().index,
  28. Eq(max_node_id.AsDesugared().AsDesugared().index));
  29. EXPECT_THAT(static_cast<LocId>(ImportIRInstId(0)).index, Eq(-2 - (1 << 25)));
  30. EXPECT_THAT(static_cast<LocId>(ImportIRInstId(ImportIRInstId::Max - 1)).index,
  31. Eq(std::numeric_limits<int32_t>::min()));
  32. }
  33. // A standard parameterized test for (is_desugared, index).
  34. class IdsTestWithParam
  35. : public testing::TestWithParam<std::tuple<bool, int32_t>> {
  36. public:
  37. explicit IdsTestWithParam() {
  38. llvm::errs() << "is_desugared=" << is_desugared() << ", index=" << index()
  39. << "\n";
  40. }
  41. // Returns IdT with its matching LocId form. Sets flags based on test
  42. // parameters.
  43. template <typename IdT>
  44. auto BuildIdAndLocId() -> std::pair<IdT, LocId> {
  45. IdT id(index());
  46. if (is_desugared()) {
  47. return {id, LocId(id).AsDesugared()};
  48. } else {
  49. return {id, LocId(id)};
  50. }
  51. }
  52. auto is_desugared() -> bool { return std::get<0>(GetParam()); }
  53. auto index() -> int32_t { return std::get<1>(GetParam()); }
  54. };
  55. // Returns a test case generator for edge-case values.
  56. static auto GetValueRange(int32_t max) -> auto {
  57. return testing::Values(0, 1, max - 2, max - 1);
  58. }
  59. // Returns a test case generator for `IdsTestWithParam` uses.
  60. static auto CombineWithFlags(auto value_range) -> auto {
  61. return testing::Combine(testing::Bool(), value_range);
  62. }
  63. class LocIdAsNoneTestWithParam : public IdsTestWithParam {};
  64. INSTANTIATE_TEST_SUITE_P(
  65. LocIdAsNoneTest, LocIdAsNoneTestWithParam,
  66. CombineWithFlags(testing::Values(Parse::NodeId::NoneIndex)));
  67. TEST_P(LocIdAsNoneTestWithParam, Test) {
  68. auto [_, loc_id] = BuildIdAndLocId<Parse::NodeId>();
  69. EXPECT_FALSE(loc_id.has_value());
  70. EXPECT_THAT(loc_id.kind(), Eq(LocId::Kind::None));
  71. EXPECT_FALSE(loc_id.is_desugared());
  72. EXPECT_THAT(loc_id.import_ir_inst_id(), Eq(ImportIRInstId::None));
  73. EXPECT_THAT(loc_id.inst_id(), Eq(InstId::None));
  74. EXPECT_THAT(loc_id.node_id(),
  75. // The actual type is NoneNodeId, so cast to NodeId.
  76. Eq<Parse::NodeId>(Parse::NodeId::None));
  77. }
  78. class LocIdAsImportIRInstIdTest : public IdsTestWithParam {};
  79. INSTANTIATE_TEST_SUITE_P(Test, LocIdAsImportIRInstIdTest,
  80. CombineWithFlags(GetValueRange(ImportIRInstId::Max)));
  81. TEST_P(LocIdAsImportIRInstIdTest, Test) {
  82. auto [import_ir_inst_id, loc_id] = BuildIdAndLocId<ImportIRInstId>();
  83. EXPECT_TRUE(loc_id.has_value());
  84. ASSERT_THAT(loc_id.kind(), Eq(LocId::Kind::ImportIRInstId));
  85. EXPECT_THAT(loc_id.import_ir_inst_id(), import_ir_inst_id);
  86. EXPECT_FALSE(loc_id.is_desugared());
  87. }
  88. class LocIdAsInstIdTest : public IdsTestWithParam {};
  89. INSTANTIATE_TEST_SUITE_P(
  90. Test, LocIdAsInstIdTest,
  91. testing::Combine(testing::Values(false),
  92. GetValueRange(std::numeric_limits<int32_t>::max())));
  93. TEST_P(LocIdAsInstIdTest, Test) {
  94. auto [inst_id, loc_id] = BuildIdAndLocId<InstId>();
  95. EXPECT_TRUE(loc_id.has_value());
  96. ASSERT_THAT(loc_id.kind(), Eq(LocId::Kind::InstId));
  97. EXPECT_THAT(loc_id.inst_id(), inst_id);
  98. // Note that `is_desugared` is invalid to use with `InstId`.
  99. }
  100. class LocIdAsNodeIdTest : public IdsTestWithParam {};
  101. INSTANTIATE_TEST_SUITE_P(Test, LocIdAsNodeIdTest,
  102. CombineWithFlags(GetValueRange(Parse::NodeId::Max)));
  103. TEST_P(LocIdAsNodeIdTest, Test) {
  104. auto [node_id, loc_id] = BuildIdAndLocId<Parse::NodeId>();
  105. EXPECT_TRUE(loc_id.has_value());
  106. ASSERT_THAT(loc_id.kind(), Eq(LocId::Kind::NodeId));
  107. EXPECT_THAT(loc_id.node_id(), node_id);
  108. EXPECT_THAT(loc_id.is_desugared(), Eq(is_desugared()));
  109. }
  110. } // namespace
  111. } // namespace Carbon::SemIR