| // Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <string.h> |
| |
| #include "gn/source_file.h" |
| |
| #include "base/logging.h" |
| #include "gn/filesystem_utils.h" |
| #include "gn/source_dir.h" |
| #include "util/build_config.h" |
| |
| namespace { |
| |
| void AssertValueSourceFileString(const std::string& s) { |
| #if defined(OS_WIN) |
| DCHECK(s[0] == '/' || |
| (s.size() > 2 && s[0] != '/' && s[1] == ':' && IsSlash(s[2]))); |
| #else |
| DCHECK(s[0] == '/'); |
| #endif |
| DCHECK(!EndsWithSlash(s)) << s; |
| } |
| |
| bool EndsWithExtension(std::string_view str, std::string_view ext) { |
| return str.size() > ext.size() && str[str.size() - ext.size() - 1] == '.' && |
| !::memcmp(str.data() + str.size() - ext.size(), ext.data(), |
| ext.size()); |
| } |
| |
| SourceFile::Type GetSourceFileType(const std::string& file) { |
| size_t size = file.size(); |
| const char* str = file.data(); |
| |
| // First, single-char extensions. |
| if (size > 2 && str[size - 2] == '.') { |
| switch (str[size - 1]) { |
| case 'c': |
| return SourceFile::SOURCE_C; // .c |
| case 'h': |
| return SourceFile::SOURCE_H; // .h |
| case 'm': |
| return SourceFile::SOURCE_M; // .m |
| case 'o': |
| return SourceFile::SOURCE_O; // .o |
| case 'S': |
| case 's': |
| return SourceFile::SOURCE_S; // .S and .s |
| default: |
| return SourceFile::SOURCE_UNKNOWN; |
| } |
| } |
| |
| // Second, two-char extensions |
| if (size > 3 && str[size - 3] == '.') { |
| #define TAG2(c1, c2) ((unsigned)(c1) | ((unsigned)(c2) << 8)) |
| switch (TAG2(str[size - 2], str[size - 1])) { |
| case TAG2('c', 'c'): |
| return SourceFile::SOURCE_CPP; // .cc |
| case TAG2('g', 'o'): |
| return SourceFile::SOURCE_GO; // .go |
| case TAG2('h', 'h'): |
| return SourceFile::SOURCE_H; // .hh |
| case TAG2('m', 'm'): |
| return SourceFile::SOURCE_MM; // .mm |
| case TAG2('r', 'c'): |
| return SourceFile::SOURCE_RC; // .rc |
| case TAG2('r', 's'): |
| return SourceFile::SOURCE_RS; // .rs |
| default: |
| return SourceFile::SOURCE_UNKNOWN; |
| } |
| #undef TAG2 |
| } |
| |
| if (size > 4 && str[size - 4] == '.') { |
| #define TAG3(c1, c2, c3) \ |
| ((unsigned)(c1) | ((unsigned)(c2) << 8) | ((unsigned)(c3) << 16)) |
| switch (TAG3(str[size - 3], str[size - 2], str[size - 1])) { |
| case TAG3('c', 'p', 'p'): |
| case TAG3('c', 'x', 'x'): |
| case TAG3('c', '+', '+'): |
| return SourceFile::SOURCE_CPP; |
| case TAG3('h', 'p', 'p'): |
| case TAG3('h', 'x', 'x'): |
| case TAG3('i', 'n', 'c'): |
| case TAG3('i', 'p', 'p'): |
| case TAG3('i', 'n', 'l'): |
| return SourceFile::SOURCE_H; |
| case TAG3('a', 's', 'm'): |
| return SourceFile::SOURCE_S; |
| case TAG3('d', 'e', 'f'): |
| return SourceFile::SOURCE_DEF; |
| case TAG3('o', 'b', 'j'): |
| return SourceFile::SOURCE_O; |
| default: |
| return SourceFile::SOURCE_UNKNOWN; |
| } |
| #undef TAG3 |
| } |
| |
| // Other cases |
| if (EndsWithExtension(file, "swift")) |
| return SourceFile::SOURCE_SWIFT; |
| |
| if (EndsWithExtension(file, "swiftmodule")) |
| return SourceFile::SOURCE_SWIFTMODULE; |
| |
| if (EndsWithExtension(file, "modulemap")) |
| return SourceFile::SOURCE_MODULEMAP; |
| |
| return SourceFile::SOURCE_UNKNOWN; |
| } |
| |
| std::string Normalized(std::string value) { |
| DCHECK(!value.empty()); |
| AssertValueSourceFileString(value); |
| NormalizePath(&value); |
| return value; |
| } |
| |
| } // namespace |
| |
| SourceFile::SourceFile(const std::string& value) |
| : SourceFile(StringAtom(Normalized(value))) {} |
| |
| SourceFile::SourceFile(std::string&& value) |
| : SourceFile(StringAtom(Normalized(std::move(value)))) {} |
| |
| SourceFile::SourceFile(StringAtom value) : value_(value) {} |
| |
| SourceFile::Type SourceFile::GetType() const { |
| return GetSourceFileType(value_.str()); |
| } |
| |
| bool SourceFile::IsDefType() const { |
| std::string_view v = value_.str(); |
| return EndsWithExtension(v, "def"); |
| } |
| |
| bool SourceFile::IsObjectType() const { |
| std::string_view v = value_.str(); |
| return EndsWithExtension(v, "o") || EndsWithExtension(v, "obj"); |
| } |
| |
| bool SourceFile::IsModuleMapType() const { |
| std::string_view v = value_.str(); |
| return EndsWithExtension(v, "modulemap"); |
| } |
| |
| bool SourceFile::IsSwiftType() const { |
| std::string_view v = value_.str(); |
| return EndsWithExtension(v, "swift"); |
| } |
| |
| bool SourceFile::IsSwiftModuleType() const { |
| std::string_view v = value_.str(); |
| return EndsWithExtension(v, "swiftmodule"); |
| } |
| |
| std::string SourceFile::GetName() const { |
| if (is_null()) |
| return std::string(); |
| |
| const std::string& value = value_.str(); |
| DCHECK(value.find('/') != std::string::npos); |
| size_t last_slash = value.rfind('/'); |
| return std::string(&value[last_slash + 1], value.size() - last_slash - 1); |
| } |
| |
| SourceDir SourceFile::GetDir() const { |
| if (is_null()) |
| return SourceDir(); |
| |
| const std::string& value = value_.str(); |
| DCHECK(value.find('/') != std::string::npos); |
| size_t last_slash = value.rfind('/'); |
| return SourceDir(value.substr(0, last_slash + 1)); |
| } |
| |
| base::FilePath SourceFile::Resolve(const base::FilePath& source_root) const { |
| return ResolvePath(value_.str(), true, source_root); |
| } |
| |
| void SourceFile::SetValue(const std::string& value) { |
| value_ = StringAtom(value); |
| } |
| |
| SourceFileTypeSet::SourceFileTypeSet() : empty_(true) { |
| memset(flags_, 0, |
| sizeof(bool) * static_cast<int>(SourceFile::SOURCE_NUMTYPES)); |
| } |
| |
| bool SourceFileTypeSet::CSourceUsed() const { |
| return empty_ || Get(SourceFile::SOURCE_CPP) || |
| Get(SourceFile::SOURCE_MODULEMAP) || Get(SourceFile::SOURCE_H) || |
| Get(SourceFile::SOURCE_C) || Get(SourceFile::SOURCE_M) || |
| Get(SourceFile::SOURCE_MM) || Get(SourceFile::SOURCE_RC) || |
| Get(SourceFile::SOURCE_S) || Get(SourceFile::SOURCE_O) || |
| Get(SourceFile::SOURCE_DEF); |
| } |
| |
| bool SourceFileTypeSet::RustSourceUsed() const { |
| return Get(SourceFile::SOURCE_RS); |
| } |
| |
| bool SourceFileTypeSet::GoSourceUsed() const { |
| return Get(SourceFile::SOURCE_GO); |
| } |
| |
| bool SourceFileTypeSet::SwiftSourceUsed() const { |
| return Get(SourceFile::SOURCE_SWIFT); |
| } |
| |
| bool SourceFileTypeSet::MixedSourceUsed() const { |
| return (1 << static_cast<int>(CSourceUsed()) |
| << static_cast<int>(RustSourceUsed()) |
| << static_cast<int>(GoSourceUsed()) |
| << static_cast<int>(SwiftSourceUsed())) > 2; |
| } |