/* Copyright 2017-2018 ccls Authors Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ==============================================================================*/ #pragma once #include #include #include #include #include #include #include namespace llvm { class StringRef; } namespace ccls { struct Matcher { struct Impl; std::unique_ptr impl; std::string pattern; Matcher(const std::string &pattern); // throw Matcher(Matcher&&) = default; ~Matcher(); bool Matches(const std::string &text) const; }; struct GroupMatch { std::vector whitelist, blacklist; GroupMatch(const std::vector &whitelist, const std::vector &blacklist); bool Matches(const std::string &text, std::string *blacklist_pattern = nullptr) const; }; uint64_t HashUsr(llvm::StringRef s); std::string LowerPathIfInsensitive(const std::string &path); // Ensures that |path| ends in a slash. void EnsureEndsInSlash(std::string &path); // Converts a file path to one that can be used as filename. // e.g. foo/bar.c => foo_bar.c std::string EscapeFileName(std::string path); std::string ResolveIfRelative(const std::string &directory, const std::string &path); std::string RealPath(const std::string &path); bool NormalizeFolder(std::string &path); std::optional LastWriteTime(const std::string &path); std::optional ReadContent(const std::string &filename); void WriteToFile(const std::string &filename, const std::string &content); int ReverseSubseqMatch(std::string_view pat, std::string_view text, int case_sensitivity); // http://stackoverflow.com/a/38140932 // // struct SomeHashKey { // std::string key1; // std::string key2; // bool key3; // }; // MAKE_HASHABLE(SomeHashKey, t.key1, t.key2, t.key3) inline void hash_combine(std::size_t &seed) {} template inline void hash_combine(std::size_t &seed, const T &v, Rest... rest) { std::hash hasher; seed ^= hasher(v) + 0x9e3779b9 + (seed << 6) + (seed >> 2); hash_combine(seed, rest...); } #define MAKE_HASHABLE(type, ...) \ namespace std { \ template <> struct hash { \ std::size_t operator()(const type &t) const { \ std::size_t ret = 0; \ ccls::hash_combine(ret, __VA_ARGS__); \ return ret; \ } \ }; \ } std::string GetDefaultResourceDirectory(); // Like std::optional, but the stored data is responsible for containing the // empty state. T should define a function `bool T::Valid()`. template class Maybe { T storage; public: constexpr Maybe() = default; Maybe(const Maybe &) = default; Maybe(std::nullopt_t) {} Maybe(const T &x) : storage(x) {} Maybe(T &&x) : storage(std::forward(x)) {} Maybe &operator=(const Maybe &) = default; Maybe &operator=(const T &x) { storage = x; return *this; } const T *operator->() const { return &storage; } T *operator->() { return &storage; } const T &operator*() const { return storage; } T &operator*() { return storage; } bool Valid() const { return storage.Valid(); } explicit operator bool() const { return Valid(); } operator std::optional() const { if (Valid()) return storage; return std::nullopt; } void operator=(std::optional &&o) { storage = o ? *o : T(); } // Does not test if has_value() bool operator==(const Maybe &o) const { return storage == o.storage; } bool operator!=(const Maybe &o) const { return !(*this == o); } }; template struct Vec { std::unique_ptr a; int s = 0; #if !(__clang__ || __GNUC__ > 7 || __GNUC__ == 7 && __GNUC_MINOR__ >= 4) // Work around a bug in GCC<7.4 that optional would not be // construtible. Vec() = default; Vec(const Vec &o) : a(std::make_unique(o.s)), s(o.s) { std::copy(o.a.get(), o.a.get() + o.s, a.get()); } Vec(Vec &&) = default; Vec &operator=(Vec &&) = default; Vec(std::unique_ptr a, int s) : a(std::move(a)), s(s) {} #endif const T *begin() const { return a.get(); } T *begin() { return a.get(); } const T *end() const { return a.get() + s; } T *end() { return a.get() + s; } int size() const { return s; } const T &operator[](size_t i) const { return a.get()[i]; } T &operator[](size_t i) { return a.get()[i]; } }; } // namespace ccls