xref: /openbsd-src/gnu/llvm/clang/lib/IndexSerialization/SerializablePathCollection.cpp (revision a9ac8606c53d55cee9c3a39778b249c51df111ef)
1*a9ac8606Spatrick //===--- SerializablePathCollection.cpp -- Index of paths -------*- C++ -*-===//
2*a9ac8606Spatrick //
3*a9ac8606Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*a9ac8606Spatrick // See https://llvm.org/LICENSE.txt for license information.
5*a9ac8606Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*a9ac8606Spatrick //
7*a9ac8606Spatrick //===----------------------------------------------------------------------===//
8*a9ac8606Spatrick 
9*a9ac8606Spatrick #include "clang/IndexSerialization/SerializablePathCollection.h"
10*a9ac8606Spatrick #include "llvm/Support/Path.h"
11*a9ac8606Spatrick 
12*a9ac8606Spatrick using namespace llvm;
13*a9ac8606Spatrick using namespace clang;
14*a9ac8606Spatrick using namespace clang::index;
15*a9ac8606Spatrick 
add(StringRef Str)16*a9ac8606Spatrick StringPool::StringOffsetSize StringPool::add(StringRef Str) {
17*a9ac8606Spatrick   const std::size_t Offset = Buffer.size();
18*a9ac8606Spatrick   Buffer += Str;
19*a9ac8606Spatrick   return StringPool::StringOffsetSize(Offset, Str.size());
20*a9ac8606Spatrick }
21*a9ac8606Spatrick 
addFilePath(RootDirKind Root,const StringPool::StringOffsetSize & Dir,StringRef Filename)22*a9ac8606Spatrick size_t PathPool::addFilePath(RootDirKind Root,
23*a9ac8606Spatrick                              const StringPool::StringOffsetSize &Dir,
24*a9ac8606Spatrick                              StringRef Filename) {
25*a9ac8606Spatrick   FilePaths.emplace_back(DirPath(Root, Dir), Paths.add(Filename));
26*a9ac8606Spatrick   return FilePaths.size() - 1;
27*a9ac8606Spatrick }
28*a9ac8606Spatrick 
addDirPath(StringRef Dir)29*a9ac8606Spatrick StringPool::StringOffsetSize PathPool::addDirPath(StringRef Dir) {
30*a9ac8606Spatrick   return Paths.add(Dir);
31*a9ac8606Spatrick }
32*a9ac8606Spatrick 
getFilePaths() const33*a9ac8606Spatrick llvm::ArrayRef<PathPool::FilePath> PathPool::getFilePaths() const {
34*a9ac8606Spatrick   return FilePaths;
35*a9ac8606Spatrick }
36*a9ac8606Spatrick 
getPaths() const37*a9ac8606Spatrick StringRef PathPool::getPaths() const { return Paths.getBuffer(); }
38*a9ac8606Spatrick 
SerializablePathCollection(StringRef CurrentWorkDir,StringRef SysRoot,llvm::StringRef OutputFile)39*a9ac8606Spatrick SerializablePathCollection::SerializablePathCollection(
40*a9ac8606Spatrick     StringRef CurrentWorkDir, StringRef SysRoot, llvm::StringRef OutputFile)
41*a9ac8606Spatrick     : WorkDir(CurrentWorkDir),
42*a9ac8606Spatrick       SysRoot(llvm::sys::path::parent_path(SysRoot).empty() ? StringRef()
43*a9ac8606Spatrick                                                             : SysRoot),
44*a9ac8606Spatrick       WorkDirPath(Paths.addDirPath(WorkDir)),
45*a9ac8606Spatrick       SysRootPath(Paths.addDirPath(SysRoot)),
46*a9ac8606Spatrick       OutputFilePath(Paths.addDirPath(OutputFile)) {}
47*a9ac8606Spatrick 
tryStoreFilePath(const FileEntry & FE)48*a9ac8606Spatrick size_t SerializablePathCollection::tryStoreFilePath(const FileEntry &FE) {
49*a9ac8606Spatrick   auto FileIt = UniqueFiles.find(&FE);
50*a9ac8606Spatrick   if (FileIt != UniqueFiles.end())
51*a9ac8606Spatrick     return FileIt->second;
52*a9ac8606Spatrick 
53*a9ac8606Spatrick   const auto Dir = tryStoreDirPath(sys::path::parent_path(FE.getName()));
54*a9ac8606Spatrick   const auto FileIdx =
55*a9ac8606Spatrick       Paths.addFilePath(Dir.Root, Dir.Path, sys::path::filename(FE.getName()));
56*a9ac8606Spatrick 
57*a9ac8606Spatrick   UniqueFiles.try_emplace(&FE, FileIdx);
58*a9ac8606Spatrick   return FileIdx;
59*a9ac8606Spatrick }
60*a9ac8606Spatrick 
tryStoreDirPath(StringRef Dir)61*a9ac8606Spatrick PathPool::DirPath SerializablePathCollection::tryStoreDirPath(StringRef Dir) {
62*a9ac8606Spatrick   // We don't want to strip separator if Dir is "/" - so we check size > 1.
63*a9ac8606Spatrick   while (Dir.size() > 1 && llvm::sys::path::is_separator(Dir.back()))
64*a9ac8606Spatrick     Dir = Dir.drop_back();
65*a9ac8606Spatrick 
66*a9ac8606Spatrick   auto DirIt = UniqueDirs.find(Dir);
67*a9ac8606Spatrick   if (DirIt != UniqueDirs.end())
68*a9ac8606Spatrick     return DirIt->second;
69*a9ac8606Spatrick 
70*a9ac8606Spatrick   const std::string OrigDir = Dir.str();
71*a9ac8606Spatrick 
72*a9ac8606Spatrick   PathPool::RootDirKind Root = PathPool::RootDirKind::Regular;
73*a9ac8606Spatrick   if (!SysRoot.empty() && Dir.startswith(SysRoot) &&
74*a9ac8606Spatrick       llvm::sys::path::is_separator(Dir[SysRoot.size()])) {
75*a9ac8606Spatrick     Root = PathPool::RootDirKind::SysRoot;
76*a9ac8606Spatrick     Dir = Dir.drop_front(SysRoot.size());
77*a9ac8606Spatrick   } else if (!WorkDir.empty() && Dir.startswith(WorkDir) &&
78*a9ac8606Spatrick              llvm::sys::path::is_separator(Dir[WorkDir.size()])) {
79*a9ac8606Spatrick     Root = PathPool::RootDirKind::CurrentWorkDir;
80*a9ac8606Spatrick     Dir = Dir.drop_front(WorkDir.size());
81*a9ac8606Spatrick   }
82*a9ac8606Spatrick 
83*a9ac8606Spatrick   if (Root != PathPool::RootDirKind::Regular) {
84*a9ac8606Spatrick     while (!Dir.empty() && llvm::sys::path::is_separator(Dir.front()))
85*a9ac8606Spatrick       Dir = Dir.drop_front();
86*a9ac8606Spatrick   }
87*a9ac8606Spatrick 
88*a9ac8606Spatrick   PathPool::DirPath Result(Root, Paths.addDirPath(Dir));
89*a9ac8606Spatrick   UniqueDirs.try_emplace(OrigDir, Result);
90*a9ac8606Spatrick   return Result;
91*a9ac8606Spatrick }
92