14df795bfSSam McCall //===- unittests/Serialization/SourceLocationEncodingTests.cpp ------------===//
24df795bfSSam McCall //
34df795bfSSam McCall // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
44df795bfSSam McCall // See https://llvm.org/LICENSE.txt for license information.
54df795bfSSam McCall // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
64df795bfSSam McCall //
74df795bfSSam McCall //===----------------------------------------------------------------------===//
84df795bfSSam McCall 
94df795bfSSam McCall #include "clang/Serialization/SourceLocationEncoding.h"
104df795bfSSam McCall 
114df795bfSSam McCall #include "gtest/gtest.h"
12*48169157SSam McCall #include <climits>
134df795bfSSam McCall 
144df795bfSSam McCall using namespace llvm;
154df795bfSSam McCall using namespace clang;
164df795bfSSam McCall 
174df795bfSSam McCall namespace {
184df795bfSSam McCall using LocSeq = SourceLocationSequence;
194df795bfSSam McCall 
204df795bfSSam McCall // Convert a single source location into encoded form and back.
214df795bfSSam McCall // If ExpectedEncoded is provided, verify the encoded value too.
224df795bfSSam McCall // Loc is the raw (in-memory) form of SourceLocation.
roundTrip(SourceLocation::UIntTy Loc,llvm::Optional<uint64_t> ExpectedEncoded=llvm::None)234df795bfSSam McCall void roundTrip(SourceLocation::UIntTy Loc,
244df795bfSSam McCall                llvm::Optional<uint64_t> ExpectedEncoded = llvm::None) {
254df795bfSSam McCall   uint64_t ActualEncoded =
264df795bfSSam McCall       SourceLocationEncoding::encode(SourceLocation::getFromRawEncoding(Loc));
274df795bfSSam McCall   if (ExpectedEncoded)
284df795bfSSam McCall     ASSERT_EQ(ActualEncoded, *ExpectedEncoded) << "Encoding " << Loc;
294df795bfSSam McCall   SourceLocation::UIntTy DecodedEncoded =
304df795bfSSam McCall       SourceLocationEncoding::decode(ActualEncoded).getRawEncoding();
314df795bfSSam McCall   ASSERT_EQ(DecodedEncoded, Loc) << "Decoding " << ActualEncoded;
324df795bfSSam McCall }
334df795bfSSam McCall 
344df795bfSSam McCall // As above, but use sequence encoding for a series of locations.
roundTrip(std::vector<SourceLocation::UIntTy> Locs,std::vector<uint64_t> ExpectedEncoded={})354df795bfSSam McCall void roundTrip(std::vector<SourceLocation::UIntTy> Locs,
364df795bfSSam McCall                std::vector<uint64_t> ExpectedEncoded = {}) {
374df795bfSSam McCall   std::vector<uint64_t> ActualEncoded;
384df795bfSSam McCall   {
394df795bfSSam McCall     LocSeq::State Seq;
404df795bfSSam McCall     for (auto L : Locs)
414df795bfSSam McCall       ActualEncoded.push_back(SourceLocationEncoding::encode(
424df795bfSSam McCall           SourceLocation::getFromRawEncoding(L), Seq));
434df795bfSSam McCall     if (!ExpectedEncoded.empty())
444df795bfSSam McCall       ASSERT_EQ(ActualEncoded, ExpectedEncoded)
454df795bfSSam McCall           << "Encoding " << testing::PrintToString(Locs);
464df795bfSSam McCall   }
474df795bfSSam McCall   std::vector<SourceLocation::UIntTy> DecodedEncoded;
484df795bfSSam McCall   {
494df795bfSSam McCall     LocSeq::State Seq;
504df795bfSSam McCall     for (auto L : ActualEncoded) {
514df795bfSSam McCall       SourceLocation Loc = SourceLocationEncoding::decode(L, Seq);
524df795bfSSam McCall       DecodedEncoded.push_back(Loc.getRawEncoding());
534df795bfSSam McCall     }
544df795bfSSam McCall     ASSERT_EQ(DecodedEncoded, Locs)
554df795bfSSam McCall         << "Decoding " << testing::PrintToString(ActualEncoded);
564df795bfSSam McCall   }
574df795bfSSam McCall }
584df795bfSSam McCall 
594df795bfSSam McCall constexpr SourceLocation::UIntTy MacroBit =
604df795bfSSam McCall     1 << (sizeof(SourceLocation::UIntTy) * CHAR_BIT - 1);
614df795bfSSam McCall constexpr SourceLocation::UIntTy Big = MacroBit >> 1;
624df795bfSSam McCall constexpr SourceLocation::UIntTy Biggest = -1;
634df795bfSSam McCall 
TEST(SourceLocationEncoding,Individual)644df795bfSSam McCall TEST(SourceLocationEncoding, Individual) {
654df795bfSSam McCall   roundTrip(1, 2);
664df795bfSSam McCall   roundTrip(100, 200);
674df795bfSSam McCall   roundTrip(MacroBit, 1);
684df795bfSSam McCall   roundTrip(MacroBit | 5, 11);
694df795bfSSam McCall   roundTrip(Big);
704df795bfSSam McCall   roundTrip(Big + 1);
714df795bfSSam McCall   roundTrip(MacroBit | Big);
724df795bfSSam McCall   roundTrip(MacroBit | Big + 1);
734df795bfSSam McCall }
744df795bfSSam McCall 
TEST(SourceLocationEncoding,Sequence)754df795bfSSam McCall TEST(SourceLocationEncoding, Sequence) {
764df795bfSSam McCall   roundTrip({1, 2, 3, 3, 2, 1},
774df795bfSSam McCall             {2, // 1
784df795bfSSam McCall              5, // +2 (+1 of non-raw)
794df795bfSSam McCall              5, // +2
804df795bfSSam McCall              1, // +0
814df795bfSSam McCall              4, // -2
824df795bfSSam McCall              4} // -2
834df795bfSSam McCall   );
844df795bfSSam McCall   roundTrip({100, 0, 100},
854df795bfSSam McCall             {200, // 100
864df795bfSSam McCall              0,   // 0
874df795bfSSam McCall              1}   // +0
884df795bfSSam McCall   );
894df795bfSSam McCall 
904df795bfSSam McCall   roundTrip({1, Big}, {2, ((Big - 1) << 2) + 1});
914df795bfSSam McCall   roundTrip({2, MacroBit | Big}, {4, ((Big - 1) << 2) - 1});
924df795bfSSam McCall 
934df795bfSSam McCall   roundTrip({3, MacroBit | 5, MacroBit | 4, 3},
944df795bfSSam McCall             {6,  // 3
954df795bfSSam McCall              11, // +5 (+2 of non-raw + set macro bit)
964df795bfSSam McCall              4,  // -2
974df795bfSSam McCall              6}  // -3 (-2 of non-raw, clear macro bit)
984df795bfSSam McCall   );
994df795bfSSam McCall 
1004df795bfSSam McCall   roundTrip(
1014df795bfSSam McCall       {123 | MacroBit, 1, 9, Biggest, Big, Big + 1, 0, MacroBit | Big, 0});
1024df795bfSSam McCall }
1034df795bfSSam McCall 
1044df795bfSSam McCall } // namespace
105