|  | //===--- Format.cpp - Format C++ code -------------------------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | /// | 
|  | /// \file | 
|  | /// This file implements functions declared in Format.h. This will be | 
|  | /// split into separate files as we go. | 
|  | /// | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "clang/Format/Format.h" | 
|  | #include "AffectedRangeManager.h" | 
|  | #include "ContinuationIndenter.h" | 
|  | #include "FormatInternal.h" | 
|  | #include "FormatTokenLexer.h" | 
|  | #include "NamespaceEndCommentsFixer.h" | 
|  | #include "SortJavaScriptImports.h" | 
|  | #include "TokenAnalyzer.h" | 
|  | #include "TokenAnnotator.h" | 
|  | #include "UnwrappedLineFormatter.h" | 
|  | #include "UnwrappedLineParser.h" | 
|  | #include "UsingDeclarationsSorter.h" | 
|  | #include "WhitespaceManager.h" | 
|  | #include "clang/Basic/Diagnostic.h" | 
|  | #include "clang/Basic/DiagnosticOptions.h" | 
|  | #include "clang/Basic/SourceManager.h" | 
|  | #include "clang/Lex/Lexer.h" | 
|  | #include "clang/Tooling/Inclusions/HeaderIncludes.h" | 
|  | #include "llvm/ADT/STLExtras.h" | 
|  | #include "llvm/ADT/StringRef.h" | 
|  | #include "llvm/Support/Allocator.h" | 
|  | #include "llvm/Support/Debug.h" | 
|  | #include "llvm/Support/Path.h" | 
|  | #include "llvm/Support/Regex.h" | 
|  | #include "llvm/Support/VirtualFileSystem.h" | 
|  | #include "llvm/Support/YAMLTraits.h" | 
|  | #include <algorithm> | 
|  | #include <memory> | 
|  | #include <mutex> | 
|  | #include <string> | 
|  | #include <unordered_map> | 
|  |  | 
|  | #define DEBUG_TYPE "format-formatter" | 
|  |  | 
|  | using clang::format::FormatStyle; | 
|  |  | 
|  | LLVM_YAML_IS_SEQUENCE_VECTOR(clang::format::FormatStyle::RawStringFormat) | 
|  |  | 
|  | namespace llvm { | 
|  | namespace yaml { | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::LanguageKind> { | 
|  | static void enumeration(IO &IO, FormatStyle::LanguageKind &Value) { | 
|  | IO.enumCase(Value, "Cpp", FormatStyle::LK_Cpp); | 
|  | IO.enumCase(Value, "Java", FormatStyle::LK_Java); | 
|  | IO.enumCase(Value, "JavaScript", FormatStyle::LK_JavaScript); | 
|  | IO.enumCase(Value, "ObjC", FormatStyle::LK_ObjC); | 
|  | IO.enumCase(Value, "Proto", FormatStyle::LK_Proto); | 
|  | IO.enumCase(Value, "TableGen", FormatStyle::LK_TableGen); | 
|  | IO.enumCase(Value, "TextProto", FormatStyle::LK_TextProto); | 
|  | IO.enumCase(Value, "CSharp", FormatStyle::LK_CSharp); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::LanguageStandard> { | 
|  | static void enumeration(IO &IO, FormatStyle::LanguageStandard &Value) { | 
|  | IO.enumCase(Value, "c++03", FormatStyle::LS_Cpp03); | 
|  | IO.enumCase(Value, "C++03", FormatStyle::LS_Cpp03); // Legacy alias | 
|  | IO.enumCase(Value, "Cpp03", FormatStyle::LS_Cpp03); // Legacy alias | 
|  |  | 
|  | IO.enumCase(Value, "c++11", FormatStyle::LS_Cpp11); | 
|  | IO.enumCase(Value, "C++11", FormatStyle::LS_Cpp11); // Legacy alias | 
|  |  | 
|  | IO.enumCase(Value, "c++14", FormatStyle::LS_Cpp14); | 
|  | IO.enumCase(Value, "c++17", FormatStyle::LS_Cpp17); | 
|  | IO.enumCase(Value, "c++20", FormatStyle::LS_Cpp20); | 
|  |  | 
|  | IO.enumCase(Value, "Latest", FormatStyle::LS_Latest); | 
|  | IO.enumCase(Value, "Cpp11", FormatStyle::LS_Latest); // Legacy alias | 
|  | IO.enumCase(Value, "Auto", FormatStyle::LS_Auto); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::UseTabStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::UseTabStyle &Value) { | 
|  | IO.enumCase(Value, "Never", FormatStyle::UT_Never); | 
|  | IO.enumCase(Value, "false", FormatStyle::UT_Never); | 
|  | IO.enumCase(Value, "Always", FormatStyle::UT_Always); | 
|  | IO.enumCase(Value, "true", FormatStyle::UT_Always); | 
|  | IO.enumCase(Value, "ForIndentation", FormatStyle::UT_ForIndentation); | 
|  | IO.enumCase(Value, "ForContinuationAndIndentation", | 
|  | FormatStyle::UT_ForContinuationAndIndentation); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::JavaScriptQuoteStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::JavaScriptQuoteStyle &Value) { | 
|  | IO.enumCase(Value, "Leave", FormatStyle::JSQS_Leave); | 
|  | IO.enumCase(Value, "Single", FormatStyle::JSQS_Single); | 
|  | IO.enumCase(Value, "Double", FormatStyle::JSQS_Double); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::ShortBlockStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::ShortBlockStyle &Value) { | 
|  | IO.enumCase(Value, "Never", FormatStyle::SBS_Never); | 
|  | IO.enumCase(Value, "false", FormatStyle::SBS_Never); | 
|  | IO.enumCase(Value, "Always", FormatStyle::SBS_Always); | 
|  | IO.enumCase(Value, "true", FormatStyle::SBS_Always); | 
|  | IO.enumCase(Value, "Empty", FormatStyle::SBS_Empty); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::ShortFunctionStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::ShortFunctionStyle &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::SFS_None); | 
|  | IO.enumCase(Value, "false", FormatStyle::SFS_None); | 
|  | IO.enumCase(Value, "All", FormatStyle::SFS_All); | 
|  | IO.enumCase(Value, "true", FormatStyle::SFS_All); | 
|  | IO.enumCase(Value, "Inline", FormatStyle::SFS_Inline); | 
|  | IO.enumCase(Value, "InlineOnly", FormatStyle::SFS_InlineOnly); | 
|  | IO.enumCase(Value, "Empty", FormatStyle::SFS_Empty); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::ShortIfStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::ShortIfStyle &Value) { | 
|  | IO.enumCase(Value, "Never", FormatStyle::SIS_Never); | 
|  | IO.enumCase(Value, "Always", FormatStyle::SIS_Always); | 
|  | IO.enumCase(Value, "WithoutElse", FormatStyle::SIS_WithoutElse); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "false", FormatStyle::SIS_Never); | 
|  | IO.enumCase(Value, "true", FormatStyle::SIS_WithoutElse); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::ShortLambdaStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::ShortLambdaStyle &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::SLS_None); | 
|  | IO.enumCase(Value, "false", FormatStyle::SLS_None); | 
|  | IO.enumCase(Value, "Empty", FormatStyle::SLS_Empty); | 
|  | IO.enumCase(Value, "Inline", FormatStyle::SLS_Inline); | 
|  | IO.enumCase(Value, "All", FormatStyle::SLS_All); | 
|  | IO.enumCase(Value, "true", FormatStyle::SLS_All); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::BinPackStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::BinPackStyle &Value) { | 
|  | IO.enumCase(Value, "Auto", FormatStyle::BPS_Auto); | 
|  | IO.enumCase(Value, "Always", FormatStyle::BPS_Always); | 
|  | IO.enumCase(Value, "Never", FormatStyle::BPS_Never); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::BinaryOperatorStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::BinaryOperatorStyle &Value) { | 
|  | IO.enumCase(Value, "All", FormatStyle::BOS_All); | 
|  | IO.enumCase(Value, "true", FormatStyle::BOS_All); | 
|  | IO.enumCase(Value, "None", FormatStyle::BOS_None); | 
|  | IO.enumCase(Value, "false", FormatStyle::BOS_None); | 
|  | IO.enumCase(Value, "NonAssignment", FormatStyle::BOS_NonAssignment); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::BraceBreakingStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::BraceBreakingStyle &Value) { | 
|  | IO.enumCase(Value, "Attach", FormatStyle::BS_Attach); | 
|  | IO.enumCase(Value, "Linux", FormatStyle::BS_Linux); | 
|  | IO.enumCase(Value, "Mozilla", FormatStyle::BS_Mozilla); | 
|  | IO.enumCase(Value, "Stroustrup", FormatStyle::BS_Stroustrup); | 
|  | IO.enumCase(Value, "Allman", FormatStyle::BS_Allman); | 
|  | IO.enumCase(Value, "Whitesmiths", FormatStyle::BS_Whitesmiths); | 
|  | IO.enumCase(Value, "GNU", FormatStyle::BS_GNU); | 
|  | IO.enumCase(Value, "WebKit", FormatStyle::BS_WebKit); | 
|  | IO.enumCase(Value, "Custom", FormatStyle::BS_Custom); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits< | 
|  | FormatStyle::BraceWrappingAfterControlStatementStyle> { | 
|  | static void | 
|  | enumeration(IO &IO, | 
|  | FormatStyle::BraceWrappingAfterControlStatementStyle &Value) { | 
|  | IO.enumCase(Value, "false", FormatStyle::BWACS_Never); | 
|  | IO.enumCase(Value, "true", FormatStyle::BWACS_Always); | 
|  | IO.enumCase(Value, "Never", FormatStyle::BWACS_Never); | 
|  | IO.enumCase(Value, "MultiLine", FormatStyle::BWACS_MultiLine); | 
|  | IO.enumCase(Value, "Always", FormatStyle::BWACS_Always); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::BreakConstructorInitializersStyle> { | 
|  | static void | 
|  | enumeration(IO &IO, FormatStyle::BreakConstructorInitializersStyle &Value) { | 
|  | IO.enumCase(Value, "BeforeColon", FormatStyle::BCIS_BeforeColon); | 
|  | IO.enumCase(Value, "BeforeComma", FormatStyle::BCIS_BeforeComma); | 
|  | IO.enumCase(Value, "AfterColon", FormatStyle::BCIS_AfterColon); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::BreakInheritanceListStyle> { | 
|  | static void enumeration(IO &IO, | 
|  | FormatStyle::BreakInheritanceListStyle &Value) { | 
|  | IO.enumCase(Value, "BeforeColon", FormatStyle::BILS_BeforeColon); | 
|  | IO.enumCase(Value, "BeforeComma", FormatStyle::BILS_BeforeComma); | 
|  | IO.enumCase(Value, "AfterColon", FormatStyle::BILS_AfterColon); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::PPDirectiveIndentStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::PPDirectiveIndentStyle &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::PPDIS_None); | 
|  | IO.enumCase(Value, "AfterHash", FormatStyle::PPDIS_AfterHash); | 
|  | IO.enumCase(Value, "BeforeHash", FormatStyle::PPDIS_BeforeHash); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::ReturnTypeBreakingStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::ReturnTypeBreakingStyle &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::RTBS_None); | 
|  | IO.enumCase(Value, "All", FormatStyle::RTBS_All); | 
|  | IO.enumCase(Value, "TopLevel", FormatStyle::RTBS_TopLevel); | 
|  | IO.enumCase(Value, "TopLevelDefinitions", | 
|  | FormatStyle::RTBS_TopLevelDefinitions); | 
|  | IO.enumCase(Value, "AllDefinitions", FormatStyle::RTBS_AllDefinitions); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::BreakTemplateDeclarationsStyle> { | 
|  | static void enumeration(IO &IO, | 
|  | FormatStyle::BreakTemplateDeclarationsStyle &Value) { | 
|  | IO.enumCase(Value, "No", FormatStyle::BTDS_No); | 
|  | IO.enumCase(Value, "MultiLine", FormatStyle::BTDS_MultiLine); | 
|  | IO.enumCase(Value, "Yes", FormatStyle::BTDS_Yes); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "false", FormatStyle::BTDS_MultiLine); | 
|  | IO.enumCase(Value, "true", FormatStyle::BTDS_Yes); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::DefinitionReturnTypeBreakingStyle> { | 
|  | static void | 
|  | enumeration(IO &IO, FormatStyle::DefinitionReturnTypeBreakingStyle &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::DRTBS_None); | 
|  | IO.enumCase(Value, "All", FormatStyle::DRTBS_All); | 
|  | IO.enumCase(Value, "TopLevel", FormatStyle::DRTBS_TopLevel); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "false", FormatStyle::DRTBS_None); | 
|  | IO.enumCase(Value, "true", FormatStyle::DRTBS_All); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::NamespaceIndentationKind> { | 
|  | static void enumeration(IO &IO, | 
|  | FormatStyle::NamespaceIndentationKind &Value) { | 
|  | IO.enumCase(Value, "None", FormatStyle::NI_None); | 
|  | IO.enumCase(Value, "Inner", FormatStyle::NI_Inner); | 
|  | IO.enumCase(Value, "All", FormatStyle::NI_All); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::BracketAlignmentStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::BracketAlignmentStyle &Value) { | 
|  | IO.enumCase(Value, "Align", FormatStyle::BAS_Align); | 
|  | IO.enumCase(Value, "DontAlign", FormatStyle::BAS_DontAlign); | 
|  | IO.enumCase(Value, "AlwaysBreak", FormatStyle::BAS_AlwaysBreak); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "true", FormatStyle::BAS_Align); | 
|  | IO.enumCase(Value, "false", FormatStyle::BAS_DontAlign); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::EscapedNewlineAlignmentStyle> { | 
|  | static void enumeration(IO &IO, | 
|  | FormatStyle::EscapedNewlineAlignmentStyle &Value) { | 
|  | IO.enumCase(Value, "DontAlign", FormatStyle::ENAS_DontAlign); | 
|  | IO.enumCase(Value, "Left", FormatStyle::ENAS_Left); | 
|  | IO.enumCase(Value, "Right", FormatStyle::ENAS_Right); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "true", FormatStyle::ENAS_Left); | 
|  | IO.enumCase(Value, "false", FormatStyle::ENAS_Right); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct ScalarEnumerationTraits<FormatStyle::PointerAlignmentStyle> { | 
|  | static void enumeration(IO &IO, FormatStyle::PointerAlignmentStyle &Value) { | 
|  | IO.enumCase(Value, "Middle", FormatStyle::PAS_Middle); | 
|  | IO.enumCase(Value, "Left", FormatStyle::PAS_Left); | 
|  | IO.enumCase(Value, "Right", FormatStyle::PAS_Right); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "true", FormatStyle::PAS_Left); | 
|  | IO.enumCase(Value, "false", FormatStyle::PAS_Right); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> | 
|  | struct ScalarEnumerationTraits<FormatStyle::SpaceBeforeParensOptions> { | 
|  | static void enumeration(IO &IO, | 
|  | FormatStyle::SpaceBeforeParensOptions &Value) { | 
|  | IO.enumCase(Value, "Never", FormatStyle::SBPO_Never); | 
|  | IO.enumCase(Value, "ControlStatements", | 
|  | FormatStyle::SBPO_ControlStatements); | 
|  | IO.enumCase(Value, "NonEmptyParentheses", | 
|  | FormatStyle::SBPO_NonEmptyParentheses); | 
|  | IO.enumCase(Value, "Always", FormatStyle::SBPO_Always); | 
|  |  | 
|  | // For backward compatibility. | 
|  | IO.enumCase(Value, "false", FormatStyle::SBPO_Never); | 
|  | IO.enumCase(Value, "true", FormatStyle::SBPO_ControlStatements); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct MappingTraits<FormatStyle> { | 
|  | static void mapping(IO &IO, FormatStyle &Style) { | 
|  | // When reading, read the language first, we need it for getPredefinedStyle. | 
|  | IO.mapOptional("Language", Style.Language); | 
|  |  | 
|  | if (IO.outputting()) { | 
|  | StringRef StylesArray[] = {"LLVM",   "Google", "Chromium", "Mozilla", | 
|  | "WebKit", "GNU",    "Microsoft"}; | 
|  | ArrayRef<StringRef> Styles(StylesArray); | 
|  | for (size_t i = 0, e = Styles.size(); i < e; ++i) { | 
|  | StringRef StyleName(Styles[i]); | 
|  | FormatStyle PredefinedStyle; | 
|  | if (getPredefinedStyle(StyleName, Style.Language, &PredefinedStyle) && | 
|  | Style == PredefinedStyle) { | 
|  | IO.mapOptional("# BasedOnStyle", StyleName); | 
|  | break; | 
|  | } | 
|  | } | 
|  | } else { | 
|  | StringRef BasedOnStyle; | 
|  | IO.mapOptional("BasedOnStyle", BasedOnStyle); | 
|  | if (!BasedOnStyle.empty()) { | 
|  | FormatStyle::LanguageKind OldLanguage = Style.Language; | 
|  | FormatStyle::LanguageKind Language = | 
|  | ((FormatStyle *)IO.getContext())->Language; | 
|  | if (!getPredefinedStyle(BasedOnStyle, Language, &Style)) { | 
|  | IO.setError(Twine("Unknown value for BasedOnStyle: ", BasedOnStyle)); | 
|  | return; | 
|  | } | 
|  | Style.Language = OldLanguage; | 
|  | } | 
|  | } | 
|  |  | 
|  | // For backward compatibility. | 
|  | if (!IO.outputting()) { | 
|  | IO.mapOptional("AlignEscapedNewlinesLeft", Style.AlignEscapedNewlines); | 
|  | IO.mapOptional("DerivePointerBinding", Style.DerivePointerAlignment); | 
|  | IO.mapOptional("IndentFunctionDeclarationAfterType", | 
|  | Style.IndentWrappedFunctionNames); | 
|  | IO.mapOptional("PointerBindsToType", Style.PointerAlignment); | 
|  | IO.mapOptional("SpaceAfterControlStatementKeyword", | 
|  | Style.SpaceBeforeParens); | 
|  | } | 
|  |  | 
|  | IO.mapOptional("AccessModifierOffset", Style.AccessModifierOffset); | 
|  | IO.mapOptional("AlignAfterOpenBracket", Style.AlignAfterOpenBracket); | 
|  | IO.mapOptional("AlignConsecutiveMacros", Style.AlignConsecutiveMacros); | 
|  | IO.mapOptional("AlignConsecutiveAssignments", | 
|  | Style.AlignConsecutiveAssignments); | 
|  | IO.mapOptional("AlignConsecutiveDeclarations", | 
|  | Style.AlignConsecutiveDeclarations); | 
|  | IO.mapOptional("AlignEscapedNewlines", Style.AlignEscapedNewlines); | 
|  | IO.mapOptional("AlignOperands", Style.AlignOperands); | 
|  | IO.mapOptional("AlignTrailingComments", Style.AlignTrailingComments); | 
|  | IO.mapOptional("AllowAllArgumentsOnNextLine", | 
|  | Style.AllowAllArgumentsOnNextLine); | 
|  | IO.mapOptional("AllowAllConstructorInitializersOnNextLine", | 
|  | Style.AllowAllConstructorInitializersOnNextLine); | 
|  | IO.mapOptional("AllowAllParametersOfDeclarationOnNextLine", | 
|  | Style.AllowAllParametersOfDeclarationOnNextLine); | 
|  | IO.mapOptional("AllowShortBlocksOnASingleLine", | 
|  | Style.AllowShortBlocksOnASingleLine); | 
|  | IO.mapOptional("AllowShortCaseLabelsOnASingleLine", | 
|  | Style.AllowShortCaseLabelsOnASingleLine); | 
|  | IO.mapOptional("AllowShortFunctionsOnASingleLine", | 
|  | Style.AllowShortFunctionsOnASingleLine); | 
|  | IO.mapOptional("AllowShortLambdasOnASingleLine", | 
|  | Style.AllowShortLambdasOnASingleLine); | 
|  | IO.mapOptional("AllowShortIfStatementsOnASingleLine", | 
|  | Style.AllowShortIfStatementsOnASingleLine); | 
|  | IO.mapOptional("AllowShortLoopsOnASingleLine", | 
|  | Style.AllowShortLoopsOnASingleLine); | 
|  | IO.mapOptional("AlwaysBreakAfterDefinitionReturnType", | 
|  | Style.AlwaysBreakAfterDefinitionReturnType); | 
|  | IO.mapOptional("AlwaysBreakAfterReturnType", | 
|  | Style.AlwaysBreakAfterReturnType); | 
|  |  | 
|  | // If AlwaysBreakAfterDefinitionReturnType was specified but | 
|  | // AlwaysBreakAfterReturnType was not, initialize the latter from the | 
|  | // former for backwards compatibility. | 
|  | if (Style.AlwaysBreakAfterDefinitionReturnType != FormatStyle::DRTBS_None && | 
|  | Style.AlwaysBreakAfterReturnType == FormatStyle::RTBS_None) { | 
|  | if (Style.AlwaysBreakAfterDefinitionReturnType == FormatStyle::DRTBS_All) | 
|  | Style.AlwaysBreakAfterReturnType = FormatStyle::RTBS_AllDefinitions; | 
|  | else if (Style.AlwaysBreakAfterDefinitionReturnType == | 
|  | FormatStyle::DRTBS_TopLevel) | 
|  | Style.AlwaysBreakAfterReturnType = | 
|  | FormatStyle::RTBS_TopLevelDefinitions; | 
|  | } | 
|  |  | 
|  | IO.mapOptional("AlwaysBreakBeforeMultilineStrings", | 
|  | Style.AlwaysBreakBeforeMultilineStrings); | 
|  | IO.mapOptional("AlwaysBreakTemplateDeclarations", | 
|  | Style.AlwaysBreakTemplateDeclarations); | 
|  | IO.mapOptional("BinPackArguments", Style.BinPackArguments); | 
|  | IO.mapOptional("BinPackParameters", Style.BinPackParameters); | 
|  | IO.mapOptional("BraceWrapping", Style.BraceWrapping); | 
|  | IO.mapOptional("BreakBeforeBinaryOperators", | 
|  | Style.BreakBeforeBinaryOperators); | 
|  | IO.mapOptional("BreakBeforeBraces", Style.BreakBeforeBraces); | 
|  |  | 
|  | bool BreakBeforeInheritanceComma = false; | 
|  | IO.mapOptional("BreakBeforeInheritanceComma", BreakBeforeInheritanceComma); | 
|  | IO.mapOptional("BreakInheritanceList", Style.BreakInheritanceList); | 
|  | // If BreakBeforeInheritanceComma was specified but | 
|  | // BreakInheritance was not, initialize the latter from the | 
|  | // former for backwards compatibility. | 
|  | if (BreakBeforeInheritanceComma && | 
|  | Style.BreakInheritanceList == FormatStyle::BILS_BeforeColon) | 
|  | Style.BreakInheritanceList = FormatStyle::BILS_BeforeComma; | 
|  |  | 
|  | IO.mapOptional("BreakBeforeTernaryOperators", | 
|  | Style.BreakBeforeTernaryOperators); | 
|  |  | 
|  | bool BreakConstructorInitializersBeforeComma = false; | 
|  | IO.mapOptional("BreakConstructorInitializersBeforeComma", | 
|  | BreakConstructorInitializersBeforeComma); | 
|  | IO.mapOptional("BreakConstructorInitializers", | 
|  | Style.BreakConstructorInitializers); | 
|  | // If BreakConstructorInitializersBeforeComma was specified but | 
|  | // BreakConstructorInitializers was not, initialize the latter from the | 
|  | // former for backwards compatibility. | 
|  | if (BreakConstructorInitializersBeforeComma && | 
|  | Style.BreakConstructorInitializers == FormatStyle::BCIS_BeforeColon) | 
|  | Style.BreakConstructorInitializers = FormatStyle::BCIS_BeforeComma; | 
|  |  | 
|  | IO.mapOptional("BreakAfterJavaFieldAnnotations", | 
|  | Style.BreakAfterJavaFieldAnnotations); | 
|  | IO.mapOptional("BreakStringLiterals", Style.BreakStringLiterals); | 
|  | IO.mapOptional("ColumnLimit", Style.ColumnLimit); | 
|  | IO.mapOptional("CommentPragmas", Style.CommentPragmas); | 
|  | IO.mapOptional("CompactNamespaces", Style.CompactNamespaces); | 
|  | IO.mapOptional("ConstructorInitializerAllOnOneLineOrOnePerLine", | 
|  | Style.ConstructorInitializerAllOnOneLineOrOnePerLine); | 
|  | IO.mapOptional("ConstructorInitializerIndentWidth", | 
|  | Style.ConstructorInitializerIndentWidth); | 
|  | IO.mapOptional("ContinuationIndentWidth", Style.ContinuationIndentWidth); | 
|  | IO.mapOptional("Cpp11BracedListStyle", Style.Cpp11BracedListStyle); | 
|  | IO.mapOptional("DerivePointerAlignment", Style.DerivePointerAlignment); | 
|  | IO.mapOptional("DisableFormat", Style.DisableFormat); | 
|  | IO.mapOptional("ExperimentalAutoDetectBinPacking", | 
|  | Style.ExperimentalAutoDetectBinPacking); | 
|  | IO.mapOptional("FixNamespaceComments", Style.FixNamespaceComments); | 
|  | IO.mapOptional("ForEachMacros", Style.ForEachMacros); | 
|  | IO.mapOptional("IncludeBlocks", Style.IncludeStyle.IncludeBlocks); | 
|  | IO.mapOptional("IncludeCategories", Style.IncludeStyle.IncludeCategories); | 
|  | IO.mapOptional("IncludeIsMainRegex", Style.IncludeStyle.IncludeIsMainRegex); | 
|  | IO.mapOptional("IncludeIsMainSourceRegex", | 
|  | Style.IncludeStyle.IncludeIsMainSourceRegex); | 
|  | IO.mapOptional("IndentCaseLabels", Style.IndentCaseLabels); | 
|  | IO.mapOptional("IndentGotoLabels", Style.IndentGotoLabels); | 
|  | IO.mapOptional("IndentPPDirectives", Style.IndentPPDirectives); | 
|  | IO.mapOptional("IndentWidth", Style.IndentWidth); | 
|  | IO.mapOptional("IndentWrappedFunctionNames", | 
|  | Style.IndentWrappedFunctionNames); | 
|  | IO.mapOptional("JavaImportGroups", Style.JavaImportGroups); | 
|  | IO.mapOptional("JavaScriptQuotes", Style.JavaScriptQuotes); | 
|  | IO.mapOptional("JavaScriptWrapImports", Style.JavaScriptWrapImports); | 
|  | IO.mapOptional("KeepEmptyLinesAtTheStartOfBlocks", | 
|  | Style.KeepEmptyLinesAtTheStartOfBlocks); | 
|  | IO.mapOptional("MacroBlockBegin", Style.MacroBlockBegin); | 
|  | IO.mapOptional("MacroBlockEnd", Style.MacroBlockEnd); | 
|  | IO.mapOptional("MaxEmptyLinesToKeep", Style.MaxEmptyLinesToKeep); | 
|  | IO.mapOptional("NamespaceIndentation", Style.NamespaceIndentation); | 
|  | IO.mapOptional("NamespaceMacros", Style.NamespaceMacros); | 
|  | IO.mapOptional("ObjCBinPackProtocolList", Style.ObjCBinPackProtocolList); | 
|  | IO.mapOptional("ObjCBlockIndentWidth", Style.ObjCBlockIndentWidth); | 
|  | IO.mapOptional("ObjCSpaceAfterProperty", Style.ObjCSpaceAfterProperty); | 
|  | IO.mapOptional("ObjCSpaceBeforeProtocolList", | 
|  | Style.ObjCSpaceBeforeProtocolList); | 
|  | IO.mapOptional("PenaltyBreakAssignment", Style.PenaltyBreakAssignment); | 
|  | IO.mapOptional("PenaltyBreakBeforeFirstCallParameter", | 
|  | Style.PenaltyBreakBeforeFirstCallParameter); | 
|  | IO.mapOptional("PenaltyBreakComment", Style.PenaltyBreakComment); | 
|  | IO.mapOptional("PenaltyBreakFirstLessLess", | 
|  | Style.PenaltyBreakFirstLessLess); | 
|  | IO.mapOptional("PenaltyBreakString", Style.PenaltyBreakString); | 
|  | IO.mapOptional("PenaltyBreakTemplateDeclaration", | 
|  | Style.PenaltyBreakTemplateDeclaration); | 
|  | IO.mapOptional("PenaltyExcessCharacter", Style.PenaltyExcessCharacter); | 
|  | IO.mapOptional("PenaltyReturnTypeOnItsOwnLine", | 
|  | Style.PenaltyReturnTypeOnItsOwnLine); | 
|  | IO.mapOptional("PointerAlignment", Style.PointerAlignment); | 
|  | IO.mapOptional("RawStringFormats", Style.RawStringFormats); | 
|  | IO.mapOptional("ReflowComments", Style.ReflowComments); | 
|  | IO.mapOptional("SortIncludes", Style.SortIncludes); | 
|  | IO.mapOptional("SortUsingDeclarations", Style.SortUsingDeclarations); | 
|  | IO.mapOptional("SpaceAfterCStyleCast", Style.SpaceAfterCStyleCast); | 
|  | IO.mapOptional("SpaceAfterLogicalNot", Style.SpaceAfterLogicalNot); | 
|  | IO.mapOptional("SpaceAfterTemplateKeyword", | 
|  | Style.SpaceAfterTemplateKeyword); | 
|  | IO.mapOptional("SpaceBeforeAssignmentOperators", | 
|  | Style.SpaceBeforeAssignmentOperators); | 
|  | IO.mapOptional("SpaceBeforeCpp11BracedList", | 
|  | Style.SpaceBeforeCpp11BracedList); | 
|  | IO.mapOptional("SpaceBeforeCtorInitializerColon", | 
|  | Style.SpaceBeforeCtorInitializerColon); | 
|  | IO.mapOptional("SpaceBeforeInheritanceColon", | 
|  | Style.SpaceBeforeInheritanceColon); | 
|  | IO.mapOptional("SpaceBeforeParens", Style.SpaceBeforeParens); | 
|  | IO.mapOptional("SpaceBeforeRangeBasedForLoopColon", | 
|  | Style.SpaceBeforeRangeBasedForLoopColon); | 
|  | IO.mapOptional("SpaceInEmptyBlock", Style.SpaceInEmptyBlock); | 
|  | IO.mapOptional("SpaceInEmptyParentheses", Style.SpaceInEmptyParentheses); | 
|  | IO.mapOptional("SpacesBeforeTrailingComments", | 
|  | Style.SpacesBeforeTrailingComments); | 
|  | IO.mapOptional("SpacesInAngles", Style.SpacesInAngles); | 
|  | IO.mapOptional("SpacesInContainerLiterals", | 
|  | Style.SpacesInContainerLiterals); | 
|  | IO.mapOptional("SpacesInCStyleCastParentheses", | 
|  | Style.SpacesInCStyleCastParentheses); | 
|  | IO.mapOptional("SpacesInParentheses", Style.SpacesInParentheses); | 
|  | IO.mapOptional("SpacesInSquareBrackets", Style.SpacesInSquareBrackets); | 
|  | IO.mapOptional("Standard", Style.Standard); | 
|  | IO.mapOptional("StatementMacros", Style.StatementMacros); | 
|  | IO.mapOptional("TabWidth", Style.TabWidth); | 
|  | IO.mapOptional("TypenameMacros", Style.TypenameMacros); | 
|  | IO.mapOptional("UseTab", Style.UseTab); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct MappingTraits<FormatStyle::BraceWrappingFlags> { | 
|  | static void mapping(IO &IO, FormatStyle::BraceWrappingFlags &Wrapping) { | 
|  | IO.mapOptional("AfterCaseLabel", Wrapping.AfterCaseLabel); | 
|  | IO.mapOptional("AfterClass", Wrapping.AfterClass); | 
|  | IO.mapOptional("AfterControlStatement", Wrapping.AfterControlStatement); | 
|  | IO.mapOptional("AfterEnum", Wrapping.AfterEnum); | 
|  | IO.mapOptional("AfterFunction", Wrapping.AfterFunction); | 
|  | IO.mapOptional("AfterNamespace", Wrapping.AfterNamespace); | 
|  | IO.mapOptional("AfterObjCDeclaration", Wrapping.AfterObjCDeclaration); | 
|  | IO.mapOptional("AfterStruct", Wrapping.AfterStruct); | 
|  | IO.mapOptional("AfterUnion", Wrapping.AfterUnion); | 
|  | IO.mapOptional("AfterExternBlock", Wrapping.AfterExternBlock); | 
|  | IO.mapOptional("BeforeCatch", Wrapping.BeforeCatch); | 
|  | IO.mapOptional("BeforeElse", Wrapping.BeforeElse); | 
|  | IO.mapOptional("IndentBraces", Wrapping.IndentBraces); | 
|  | IO.mapOptional("SplitEmptyFunction", Wrapping.SplitEmptyFunction); | 
|  | IO.mapOptional("SplitEmptyRecord", Wrapping.SplitEmptyRecord); | 
|  | IO.mapOptional("SplitEmptyNamespace", Wrapping.SplitEmptyNamespace); | 
|  | } | 
|  | }; | 
|  |  | 
|  | template <> struct MappingTraits<FormatStyle::RawStringFormat> { | 
|  | static void mapping(IO &IO, FormatStyle::RawStringFormat &Format) { | 
|  | IO.mapOptional("Language", Format.Language); | 
|  | IO.mapOptional("Delimiters", Format.Delimiters); | 
|  | IO.mapOptional("EnclosingFunctions", Format.EnclosingFunctions); | 
|  | IO.mapOptional("CanonicalDelimiter", Format.CanonicalDelimiter); | 
|  | IO.mapOptional("BasedOnStyle", Format.BasedOnStyle); | 
|  | } | 
|  | }; | 
|  |  | 
|  | // Allows to read vector<FormatStyle> while keeping default values. | 
|  | // IO.getContext() should contain a pointer to the FormatStyle structure, that | 
|  | // will be used to get default values for missing keys. | 
|  | // If the first element has no Language specified, it will be treated as the | 
|  | // default one for the following elements. | 
|  | template <> struct DocumentListTraits<std::vector<FormatStyle>> { | 
|  | static size_t size(IO &IO, std::vector<FormatStyle> &Seq) { | 
|  | return Seq.size(); | 
|  | } | 
|  | static FormatStyle &element(IO &IO, std::vector<FormatStyle> &Seq, | 
|  | size_t Index) { | 
|  | if (Index >= Seq.size()) { | 
|  | assert(Index == Seq.size()); | 
|  | FormatStyle Template; | 
|  | if (!Seq.empty() && Seq[0].Language == FormatStyle::LK_None) { | 
|  | Template = Seq[0]; | 
|  | } else { | 
|  | Template = *((const FormatStyle *)IO.getContext()); | 
|  | Template.Language = FormatStyle::LK_None; | 
|  | } | 
|  | Seq.resize(Index + 1, Template); | 
|  | } | 
|  | return Seq[Index]; | 
|  | } | 
|  | }; | 
|  | } // namespace yaml | 
|  | } // namespace llvm | 
|  |  | 
|  | namespace clang { | 
|  | namespace format { | 
|  |  | 
|  | const std::error_category &getParseCategory() { | 
|  | static const ParseErrorCategory C{}; | 
|  | return C; | 
|  | } | 
|  | std::error_code make_error_code(ParseError e) { | 
|  | return std::error_code(static_cast<int>(e), getParseCategory()); | 
|  | } | 
|  |  | 
|  | inline llvm::Error make_string_error(const llvm::Twine &Message) { | 
|  | return llvm::make_error<llvm::StringError>(Message, | 
|  | llvm::inconvertibleErrorCode()); | 
|  | } | 
|  |  | 
|  | const char *ParseErrorCategory::name() const noexcept { | 
|  | return "clang-format.parse_error"; | 
|  | } | 
|  |  | 
|  | std::string ParseErrorCategory::message(int EV) const { | 
|  | switch (static_cast<ParseError>(EV)) { | 
|  | case ParseError::Success: | 
|  | return "Success"; | 
|  | case ParseError::Error: | 
|  | return "Invalid argument"; | 
|  | case ParseError::Unsuitable: | 
|  | return "Unsuitable"; | 
|  | } | 
|  | llvm_unreachable("unexpected parse error"); | 
|  | } | 
|  |  | 
|  | static FormatStyle expandPresets(const FormatStyle &Style) { | 
|  | if (Style.BreakBeforeBraces == FormatStyle::BS_Custom) | 
|  | return Style; | 
|  | FormatStyle Expanded = Style; | 
|  | Expanded.BraceWrapping = {false, false, FormatStyle::BWACS_Never, | 
|  | false, false, false, | 
|  | false, false, false, | 
|  | false, false, false, | 
|  | false, true,  true, | 
|  | true}; | 
|  | switch (Style.BreakBeforeBraces) { | 
|  | case FormatStyle::BS_Linux: | 
|  | Expanded.BraceWrapping.AfterClass = true; | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | Expanded.BraceWrapping.AfterNamespace = true; | 
|  | break; | 
|  | case FormatStyle::BS_Mozilla: | 
|  | Expanded.BraceWrapping.AfterClass = true; | 
|  | Expanded.BraceWrapping.AfterEnum = true; | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | Expanded.BraceWrapping.AfterStruct = true; | 
|  | Expanded.BraceWrapping.AfterUnion = true; | 
|  | Expanded.BraceWrapping.AfterExternBlock = true; | 
|  | Expanded.BraceWrapping.SplitEmptyFunction = true; | 
|  | Expanded.BraceWrapping.SplitEmptyRecord = false; | 
|  | break; | 
|  | case FormatStyle::BS_Stroustrup: | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | Expanded.BraceWrapping.BeforeCatch = true; | 
|  | Expanded.BraceWrapping.BeforeElse = true; | 
|  | break; | 
|  | case FormatStyle::BS_Allman: | 
|  | Expanded.BraceWrapping.AfterCaseLabel = true; | 
|  | Expanded.BraceWrapping.AfterClass = true; | 
|  | Expanded.BraceWrapping.AfterControlStatement = FormatStyle::BWACS_Always; | 
|  | Expanded.BraceWrapping.AfterEnum = true; | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | Expanded.BraceWrapping.AfterNamespace = true; | 
|  | Expanded.BraceWrapping.AfterObjCDeclaration = true; | 
|  | Expanded.BraceWrapping.AfterStruct = true; | 
|  | Expanded.BraceWrapping.AfterUnion = true; | 
|  | Expanded.BraceWrapping.AfterExternBlock = true; | 
|  | Expanded.BraceWrapping.BeforeCatch = true; | 
|  | Expanded.BraceWrapping.BeforeElse = true; | 
|  | break; | 
|  | case FormatStyle::BS_Whitesmiths: | 
|  | Expanded.BraceWrapping.AfterCaseLabel = true; | 
|  | Expanded.BraceWrapping.AfterClass = true; | 
|  | Expanded.BraceWrapping.AfterControlStatement = FormatStyle::BWACS_Always; | 
|  | Expanded.BraceWrapping.AfterEnum = true; | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | Expanded.BraceWrapping.AfterNamespace = true; | 
|  | Expanded.BraceWrapping.AfterObjCDeclaration = true; | 
|  | Expanded.BraceWrapping.AfterStruct = true; | 
|  | Expanded.BraceWrapping.AfterExternBlock = true; | 
|  | Expanded.BraceWrapping.BeforeCatch = true; | 
|  | Expanded.BraceWrapping.BeforeElse = true; | 
|  | break; | 
|  | case FormatStyle::BS_GNU: | 
|  | Expanded.BraceWrapping = {true, true, FormatStyle::BWACS_Always, | 
|  | true, true, true, | 
|  | true, true, true, | 
|  | true, true, true, | 
|  | true, true, true, | 
|  | true}; | 
|  | break; | 
|  | case FormatStyle::BS_WebKit: | 
|  | Expanded.BraceWrapping.AfterFunction = true; | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | return Expanded; | 
|  | } | 
|  |  | 
|  | FormatStyle getLLVMStyle(FormatStyle::LanguageKind Language) { | 
|  | FormatStyle LLVMStyle; | 
|  | LLVMStyle.Language = Language; | 
|  | LLVMStyle.AccessModifierOffset = -2; | 
|  | LLVMStyle.AlignEscapedNewlines = FormatStyle::ENAS_Right; | 
|  | LLVMStyle.AlignAfterOpenBracket = FormatStyle::BAS_Align; | 
|  | LLVMStyle.AlignOperands = true; | 
|  | LLVMStyle.AlignTrailingComments = true; | 
|  | LLVMStyle.AlignConsecutiveAssignments = false; | 
|  | LLVMStyle.AlignConsecutiveDeclarations = false; | 
|  | LLVMStyle.AlignConsecutiveMacros = false; | 
|  | LLVMStyle.AllowAllArgumentsOnNextLine = true; | 
|  | LLVMStyle.AllowAllConstructorInitializersOnNextLine = true; | 
|  | LLVMStyle.AllowAllParametersOfDeclarationOnNextLine = true; | 
|  | LLVMStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_All; | 
|  | LLVMStyle.AllowShortBlocksOnASingleLine = FormatStyle::SBS_Never; | 
|  | LLVMStyle.AllowShortCaseLabelsOnASingleLine = false; | 
|  | LLVMStyle.AllowShortIfStatementsOnASingleLine = FormatStyle::SIS_Never; | 
|  | LLVMStyle.AllowShortLambdasOnASingleLine = FormatStyle::SLS_All; | 
|  | LLVMStyle.AllowShortLoopsOnASingleLine = false; | 
|  | LLVMStyle.AlwaysBreakAfterReturnType = FormatStyle::RTBS_None; | 
|  | LLVMStyle.AlwaysBreakAfterDefinitionReturnType = FormatStyle::DRTBS_None; | 
|  | LLVMStyle.AlwaysBreakBeforeMultilineStrings = false; | 
|  | LLVMStyle.AlwaysBreakTemplateDeclarations = FormatStyle::BTDS_MultiLine; | 
|  | LLVMStyle.BinPackArguments = true; | 
|  | LLVMStyle.BinPackParameters = true; | 
|  | LLVMStyle.BreakBeforeBinaryOperators = FormatStyle::BOS_None; | 
|  | LLVMStyle.BreakBeforeTernaryOperators = true; | 
|  | LLVMStyle.BreakBeforeBraces = FormatStyle::BS_Attach; | 
|  | LLVMStyle.BraceWrapping = {false, false, FormatStyle::BWACS_Never, | 
|  | false, false, false, | 
|  | false, false, false, | 
|  | false, false, false, | 
|  | false, true,  true, | 
|  | true}; | 
|  | LLVMStyle.BreakAfterJavaFieldAnnotations = false; | 
|  | LLVMStyle.BreakConstructorInitializers = FormatStyle::BCIS_BeforeColon; | 
|  | LLVMStyle.BreakInheritanceList = FormatStyle::BILS_BeforeColon; | 
|  | LLVMStyle.BreakStringLiterals = true; | 
|  | LLVMStyle.ColumnLimit = 80; | 
|  | LLVMStyle.CommentPragmas = "^ IWYU pragma:"; | 
|  | LLVMStyle.CompactNamespaces = false; | 
|  | LLVMStyle.ConstructorInitializerAllOnOneLineOrOnePerLine = false; | 
|  | LLVMStyle.ConstructorInitializerIndentWidth = 4; | 
|  | LLVMStyle.ContinuationIndentWidth = 4; | 
|  | LLVMStyle.Cpp11BracedListStyle = true; | 
|  | LLVMStyle.DerivePointerAlignment = false; | 
|  | LLVMStyle.ExperimentalAutoDetectBinPacking = false; | 
|  | LLVMStyle.FixNamespaceComments = true; | 
|  | LLVMStyle.ForEachMacros.push_back("foreach"); | 
|  | LLVMStyle.ForEachMacros.push_back("Q_FOREACH"); | 
|  | LLVMStyle.ForEachMacros.push_back("BOOST_FOREACH"); | 
|  | LLVMStyle.IncludeStyle.IncludeCategories = { | 
|  | {"^\"(llvm|llvm-c|clang|clang-c)/", 2, 0}, | 
|  | {"^(<|\"(gtest|gmock|isl|json)/)", 3, 0}, | 
|  | {".*", 1, 0}}; | 
|  | LLVMStyle.IncludeStyle.IncludeIsMainRegex = "(Test)?$"; | 
|  | LLVMStyle.IncludeStyle.IncludeBlocks = tooling::IncludeStyle::IBS_Preserve; | 
|  | LLVMStyle.IndentCaseLabels = false; | 
|  | LLVMStyle.IndentGotoLabels = true; | 
|  | LLVMStyle.IndentPPDirectives = FormatStyle::PPDIS_None; | 
|  | LLVMStyle.IndentWrappedFunctionNames = false; | 
|  | LLVMStyle.IndentWidth = 2; | 
|  | LLVMStyle.JavaScriptQuotes = FormatStyle::JSQS_Leave; | 
|  | LLVMStyle.JavaScriptWrapImports = true; | 
|  | LLVMStyle.TabWidth = 8; | 
|  | LLVMStyle.MaxEmptyLinesToKeep = 1; | 
|  | LLVMStyle.KeepEmptyLinesAtTheStartOfBlocks = true; | 
|  | LLVMStyle.NamespaceIndentation = FormatStyle::NI_None; | 
|  | LLVMStyle.ObjCBinPackProtocolList = FormatStyle::BPS_Auto; | 
|  | LLVMStyle.ObjCBlockIndentWidth = 2; | 
|  | LLVMStyle.ObjCSpaceAfterProperty = false; | 
|  | LLVMStyle.ObjCSpaceBeforeProtocolList = true; | 
|  | LLVMStyle.PointerAlignment = FormatStyle::PAS_Right; | 
|  | LLVMStyle.SpacesBeforeTrailingComments = 1; | 
|  | LLVMStyle.Standard = FormatStyle::LS_Latest; | 
|  | LLVMStyle.UseTab = FormatStyle::UT_Never; | 
|  | LLVMStyle.ReflowComments = true; | 
|  | LLVMStyle.SpacesInParentheses = false; | 
|  | LLVMStyle.SpacesInSquareBrackets = false; | 
|  | LLVMStyle.SpaceInEmptyBlock = false; | 
|  | LLVMStyle.SpaceInEmptyParentheses = false; | 
|  | LLVMStyle.SpacesInContainerLiterals = true; | 
|  | LLVMStyle.SpacesInCStyleCastParentheses = false; | 
|  | LLVMStyle.SpaceAfterCStyleCast = false; | 
|  | LLVMStyle.SpaceAfterLogicalNot = false; | 
|  | LLVMStyle.SpaceAfterTemplateKeyword = true; | 
|  | LLVMStyle.SpaceBeforeCtorInitializerColon = true; | 
|  | LLVMStyle.SpaceBeforeInheritanceColon = true; | 
|  | LLVMStyle.SpaceBeforeParens = FormatStyle::SBPO_ControlStatements; | 
|  | LLVMStyle.SpaceBeforeRangeBasedForLoopColon = true; | 
|  | LLVMStyle.SpaceBeforeAssignmentOperators = true; | 
|  | LLVMStyle.SpaceBeforeCpp11BracedList = false; | 
|  | LLVMStyle.SpacesInAngles = false; | 
|  |  | 
|  | LLVMStyle.PenaltyBreakAssignment = prec::Assignment; | 
|  | LLVMStyle.PenaltyBreakComment = 300; | 
|  | LLVMStyle.PenaltyBreakFirstLessLess = 120; | 
|  | LLVMStyle.PenaltyBreakString = 1000; | 
|  | LLVMStyle.PenaltyExcessCharacter = 1000000; | 
|  | LLVMStyle.PenaltyReturnTypeOnItsOwnLine = 60; | 
|  | LLVMStyle.PenaltyBreakBeforeFirstCallParameter = 19; | 
|  | LLVMStyle.PenaltyBreakTemplateDeclaration = prec::Relational; | 
|  |  | 
|  | LLVMStyle.DisableFormat = false; | 
|  | LLVMStyle.SortIncludes = true; | 
|  | LLVMStyle.SortUsingDeclarations = true; | 
|  | LLVMStyle.StatementMacros.push_back("Q_UNUSED"); | 
|  | LLVMStyle.StatementMacros.push_back("QT_REQUIRE_VERSION"); | 
|  |  | 
|  | // Defaults that differ when not C++. | 
|  | if (Language == FormatStyle::LK_TableGen) { | 
|  | LLVMStyle.SpacesInContainerLiterals = false; | 
|  | } | 
|  |  | 
|  | return LLVMStyle; | 
|  | } | 
|  |  | 
|  | FormatStyle getGoogleStyle(FormatStyle::LanguageKind Language) { | 
|  | if (Language == FormatStyle::LK_TextProto) { | 
|  | FormatStyle GoogleStyle = getGoogleStyle(FormatStyle::LK_Proto); | 
|  | GoogleStyle.Language = FormatStyle::LK_TextProto; | 
|  |  | 
|  | return GoogleStyle; | 
|  | } | 
|  |  | 
|  | FormatStyle GoogleStyle = getLLVMStyle(Language); | 
|  |  | 
|  | GoogleStyle.AccessModifierOffset = -1; | 
|  | GoogleStyle.AlignEscapedNewlines = FormatStyle::ENAS_Left; | 
|  | GoogleStyle.AllowShortIfStatementsOnASingleLine = | 
|  | FormatStyle::SIS_WithoutElse; | 
|  | GoogleStyle.AllowShortLoopsOnASingleLine = true; | 
|  | GoogleStyle.AlwaysBreakBeforeMultilineStrings = true; | 
|  | GoogleStyle.AlwaysBreakTemplateDeclarations = FormatStyle::BTDS_Yes; | 
|  | GoogleStyle.ConstructorInitializerAllOnOneLineOrOnePerLine = true; | 
|  | GoogleStyle.DerivePointerAlignment = true; | 
|  | GoogleStyle.IncludeStyle.IncludeCategories = {{"^<ext/.*\\.h>", 2, 0}, | 
|  | {"^<.*\\.h>", 1, 0}, | 
|  | {"^<.*", 2, 0}, | 
|  | {".*", 3, 0}}; | 
|  | GoogleStyle.IncludeStyle.IncludeIsMainRegex = "([-_](test|unittest))?$"; | 
|  | GoogleStyle.IncludeStyle.IncludeBlocks = tooling::IncludeStyle::IBS_Regroup; | 
|  | GoogleStyle.IndentCaseLabels = true; | 
|  | GoogleStyle.KeepEmptyLinesAtTheStartOfBlocks = false; | 
|  | GoogleStyle.ObjCBinPackProtocolList = FormatStyle::BPS_Never; | 
|  | GoogleStyle.ObjCSpaceAfterProperty = false; | 
|  | GoogleStyle.ObjCSpaceBeforeProtocolList = true; | 
|  | GoogleStyle.PointerAlignment = FormatStyle::PAS_Left; | 
|  | GoogleStyle.RawStringFormats = { | 
|  | { | 
|  | FormatStyle::LK_Cpp, | 
|  | /*Delimiters=*/ | 
|  | { | 
|  | "cc", | 
|  | "CC", | 
|  | "cpp", | 
|  | "Cpp", | 
|  | "CPP", | 
|  | "c++", | 
|  | "C++", | 
|  | }, | 
|  | /*EnclosingFunctionNames=*/ | 
|  | {}, | 
|  | /*CanonicalDelimiter=*/"", | 
|  | /*BasedOnStyle=*/"google", | 
|  | }, | 
|  | { | 
|  | FormatStyle::LK_TextProto, | 
|  | /*Delimiters=*/ | 
|  | { | 
|  | "pb", | 
|  | "PB", | 
|  | "proto", | 
|  | "PROTO", | 
|  | }, | 
|  | /*EnclosingFunctionNames=*/ | 
|  | { | 
|  | "EqualsProto", | 
|  | "EquivToProto", | 
|  | "PARSE_PARTIAL_TEXT_PROTO", | 
|  | "PARSE_TEST_PROTO", | 
|  | "PARSE_TEXT_PROTO", | 
|  | "ParseTextOrDie", | 
|  | "ParseTextProtoOrDie", | 
|  | }, | 
|  | /*CanonicalDelimiter=*/"", | 
|  | /*BasedOnStyle=*/"google", | 
|  | }, | 
|  | }; | 
|  | GoogleStyle.SpacesBeforeTrailingComments = 2; | 
|  | GoogleStyle.Standard = FormatStyle::LS_Auto; | 
|  |  | 
|  | GoogleStyle.PenaltyReturnTypeOnItsOwnLine = 200; | 
|  | GoogleStyle.PenaltyBreakBeforeFirstCallParameter = 1; | 
|  |  | 
|  | if (Language == FormatStyle::LK_Java) { | 
|  | GoogleStyle.AlignAfterOpenBracket = FormatStyle::BAS_DontAlign; | 
|  | GoogleStyle.AlignOperands = false; | 
|  | GoogleStyle.AlignTrailingComments = false; | 
|  | GoogleStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_Empty; | 
|  | GoogleStyle.AllowShortIfStatementsOnASingleLine = FormatStyle::SIS_Never; | 
|  | GoogleStyle.AlwaysBreakBeforeMultilineStrings = false; | 
|  | GoogleStyle.BreakBeforeBinaryOperators = FormatStyle::BOS_NonAssignment; | 
|  | GoogleStyle.ColumnLimit = 100; | 
|  | GoogleStyle.SpaceAfterCStyleCast = true; | 
|  | GoogleStyle.SpacesBeforeTrailingComments = 1; | 
|  | } else if (Language == FormatStyle::LK_JavaScript) { | 
|  | GoogleStyle.AlignAfterOpenBracket = FormatStyle::BAS_AlwaysBreak; | 
|  | GoogleStyle.AlignOperands = false; | 
|  | GoogleStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_Empty; | 
|  | GoogleStyle.AlwaysBreakBeforeMultilineStrings = false; | 
|  | GoogleStyle.BreakBeforeTernaryOperators = false; | 
|  | // taze:, triple slash directives (`/// <...`), @see, which is commonly | 
|  | // followed by overlong URLs. | 
|  | GoogleStyle.CommentPragmas = "(taze:|^/[ \t]*<|@see)"; | 
|  | GoogleStyle.MaxEmptyLinesToKeep = 3; | 
|  | GoogleStyle.NamespaceIndentation = FormatStyle::NI_All; | 
|  | GoogleStyle.SpacesInContainerLiterals = false; | 
|  | GoogleStyle.JavaScriptQuotes = FormatStyle::JSQS_Single; | 
|  | GoogleStyle.JavaScriptWrapImports = false; | 
|  | } else if (Language == FormatStyle::LK_Proto) { | 
|  | GoogleStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_Empty; | 
|  | GoogleStyle.AlwaysBreakBeforeMultilineStrings = false; | 
|  | GoogleStyle.SpacesInContainerLiterals = false; | 
|  | GoogleStyle.Cpp11BracedListStyle = false; | 
|  | // This affects protocol buffer options specifications and text protos. | 
|  | // Text protos are currently mostly formatted inside C++ raw string literals | 
|  | // and often the current breaking behavior of string literals is not | 
|  | // beneficial there. Investigate turning this on once proper string reflow | 
|  | // has been implemented. | 
|  | GoogleStyle.BreakStringLiterals = false; | 
|  | } else if (Language == FormatStyle::LK_ObjC) { | 
|  | GoogleStyle.AlwaysBreakBeforeMultilineStrings = false; | 
|  | GoogleStyle.ColumnLimit = 100; | 
|  | // "Regroup" doesn't work well for ObjC yet (main header heuristic, | 
|  | // relationship between ObjC standard library headers and other heades, | 
|  | // #imports, etc.) | 
|  | GoogleStyle.IncludeStyle.IncludeBlocks = | 
|  | tooling::IncludeStyle::IBS_Preserve; | 
|  | } | 
|  |  | 
|  | return GoogleStyle; | 
|  | } | 
|  |  | 
|  | FormatStyle getChromiumStyle(FormatStyle::LanguageKind Language) { | 
|  | FormatStyle ChromiumStyle = getGoogleStyle(Language); | 
|  |  | 
|  | // Disable include reordering across blocks in Chromium code. | 
|  | // - clang-format tries to detect that foo.h is the "main" header for | 
|  | //   foo.cc and foo_unittest.cc via IncludeIsMainRegex. However, Chromium | 
|  | //   uses many other suffices (_win.cc, _mac.mm, _posix.cc, _browsertest.cc, | 
|  | //   _private.cc, _impl.cc etc) in different permutations | 
|  | //   (_win_browsertest.cc) so disable this until IncludeIsMainRegex has a | 
|  | //   better default for Chromium code. | 
|  | // - The default for .cc and .mm files is different (r357695) for Google style | 
|  | //   for the same reason. The plan is to unify this again once the main | 
|  | //   header detection works for Google's ObjC code, but this hasn't happened | 
|  | //   yet. Since Chromium has some ObjC code, switching Chromium is blocked | 
|  | //   on that. | 
|  | // - Finally, "If include reordering is harmful, put things in different | 
|  | //   blocks to prevent it" has been a recommendation for a long time that | 
|  | //   people are used to. We'll need a dev education push to change this to | 
|  | //   "If include reordering is harmful, put things in a different block and | 
|  | //   _prepend that with a comment_ to prevent it" before changing behavior. | 
|  | ChromiumStyle.IncludeStyle.IncludeBlocks = | 
|  | tooling::IncludeStyle::IBS_Preserve; | 
|  |  | 
|  | if (Language == FormatStyle::LK_Java) { | 
|  | ChromiumStyle.AllowShortIfStatementsOnASingleLine = | 
|  | FormatStyle::SIS_WithoutElse; | 
|  | ChromiumStyle.BreakAfterJavaFieldAnnotations = true; | 
|  | ChromiumStyle.ContinuationIndentWidth = 8; | 
|  | ChromiumStyle.IndentWidth = 4; | 
|  | // See styleguide for import groups: | 
|  | // https://chromium.googlesource.com/chromium/src/+/master/styleguide/java/java.md#Import-Order | 
|  | ChromiumStyle.JavaImportGroups = { | 
|  | "android", | 
|  | "androidx", | 
|  | "com", | 
|  | "dalvik", | 
|  | "junit", | 
|  | "org", | 
|  | "com.google.android.apps.chrome", | 
|  | "org.chromium", | 
|  | "java", | 
|  | "javax", | 
|  | }; | 
|  | ChromiumStyle.SortIncludes = true; | 
|  | } else if (Language == FormatStyle::LK_JavaScript) { | 
|  | ChromiumStyle.AllowShortIfStatementsOnASingleLine = FormatStyle::SIS_Never; | 
|  | ChromiumStyle.AllowShortLoopsOnASingleLine = false; | 
|  | } else { | 
|  | ChromiumStyle.AllowAllParametersOfDeclarationOnNextLine = false; | 
|  | ChromiumStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_Inline; | 
|  | ChromiumStyle.AllowShortIfStatementsOnASingleLine = FormatStyle::SIS_Never; | 
|  | ChromiumStyle.AllowShortLoopsOnASingleLine = false; | 
|  | ChromiumStyle.BinPackParameters = false; | 
|  | ChromiumStyle.DerivePointerAlignment = false; | 
|  | if (Language == FormatStyle::LK_ObjC) | 
|  | ChromiumStyle.ColumnLimit = 80; | 
|  | } | 
|  | return ChromiumStyle; | 
|  | } | 
|  |  | 
|  | FormatStyle getMozillaStyle() { | 
|  | FormatStyle MozillaStyle = getLLVMStyle(); | 
|  | MozillaStyle.AllowAllParametersOfDeclarationOnNextLine = false; | 
|  | MozillaStyle.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_Inline; | 
|  | MozillaStyle.AlwaysBreakAfterReturnType = FormatStyle::RTBS_TopLevel; | 
|  | MozillaStyle.AlwaysBreakAfterDefinitionReturnType = | 
|  | FormatStyle::DRTBS_TopLevel; | 
|  | MozillaStyle.AlwaysBreakTemplateDeclarations = FormatStyle::BTDS_Yes; | 
|  | MozillaStyle.BinPackParameters = false; | 
|  | MozillaStyle.BinPackArguments = false; | 
|  | MozillaStyle.BreakBeforeBraces = FormatStyle::BS_Mozilla; | 
|  | MozillaStyle.BreakConstructorInitializers = FormatStyle::BCIS_BeforeComma; | 
|  | MozillaStyle.BreakInheritanceList = FormatStyle::BILS_BeforeComma; | 
|  | MozillaStyle.ConstructorInitializerIndentWidth = 2; | 
|  | MozillaStyle.ContinuationIndentWidth = 2; | 
|  | MozillaStyle.Cpp11BracedListStyle = false; | 
|  | MozillaStyle.FixNamespaceComments = false; | 
|  | MozillaStyle.IndentCaseLabels = true; | 
|  | MozillaStyle.ObjCSpaceAfterProperty = true; | 
|  | MozillaStyle.ObjCSpaceBeforeProtocolList = false; | 
|  | MozillaStyle.PenaltyReturnTypeOnItsOwnLine = 200; | 
|  | MozillaStyle.PointerAlignment = FormatStyle::PAS_Left; | 
|  | MozillaStyle.SpaceAfterTemplateKeyword = false; | 
|  | return MozillaStyle; | 
|  | } | 
|  |  | 
|  | FormatStyle getWebKitStyle() { | 
|  | FormatStyle Style = getLLVMStyle(); | 
|  | Style.AccessModifierOffset = -4; | 
|  | Style.AlignAfterOpenBracket = FormatStyle::BAS_DontAlign; | 
|  | Style.AlignOperands = false; | 
|  | Style.AlignTrailingComments = false; | 
|  | Style.AllowShortBlocksOnASingleLine = FormatStyle::SBS_Empty; | 
|  | Style.BreakBeforeBinaryOperators = FormatStyle::BOS_All; | 
|  | Style.BreakBeforeBraces = FormatStyle::BS_WebKit; | 
|  | Style.BreakConstructorInitializers = FormatStyle::BCIS_BeforeComma; | 
|  | Style.Cpp11BracedListStyle = false; | 
|  | Style.ColumnLimit = 0; | 
|  | Style.FixNamespaceComments = false; | 
|  | Style.IndentWidth = 4; | 
|  | Style.NamespaceIndentation = FormatStyle::NI_Inner; | 
|  | Style.ObjCBlockIndentWidth = 4; | 
|  | Style.ObjCSpaceAfterProperty = true; | 
|  | Style.PointerAlignment = FormatStyle::PAS_Left; | 
|  | Style.SpaceBeforeCpp11BracedList = true; | 
|  | Style.SpaceInEmptyBlock = true; | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | FormatStyle getGNUStyle() { | 
|  | FormatStyle Style = getLLVMStyle(); | 
|  | Style.AlwaysBreakAfterDefinitionReturnType = FormatStyle::DRTBS_All; | 
|  | Style.AlwaysBreakAfterReturnType = FormatStyle::RTBS_AllDefinitions; | 
|  | Style.BreakBeforeBinaryOperators = FormatStyle::BOS_All; | 
|  | Style.BreakBeforeBraces = FormatStyle::BS_GNU; | 
|  | Style.BreakBeforeTernaryOperators = true; | 
|  | Style.Cpp11BracedListStyle = false; | 
|  | Style.ColumnLimit = 79; | 
|  | Style.FixNamespaceComments = false; | 
|  | Style.SpaceBeforeParens = FormatStyle::SBPO_Always; | 
|  | Style.Standard = FormatStyle::LS_Cpp03; | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | FormatStyle getMicrosoftStyle(FormatStyle::LanguageKind Language) { | 
|  | FormatStyle Style = getLLVMStyle(Language); | 
|  | Style.ColumnLimit = 120; | 
|  | Style.TabWidth = 4; | 
|  | Style.IndentWidth = 4; | 
|  | Style.UseTab = FormatStyle::UT_Never; | 
|  | Style.BreakBeforeBraces = FormatStyle::BS_Custom; | 
|  | Style.BraceWrapping.AfterClass = true; | 
|  | Style.BraceWrapping.AfterControlStatement = FormatStyle::BWACS_Always; | 
|  | Style.BraceWrapping.AfterEnum = true; | 
|  | Style.BraceWrapping.AfterFunction = true; | 
|  | Style.BraceWrapping.AfterNamespace = true; | 
|  | Style.BraceWrapping.AfterObjCDeclaration = true; | 
|  | Style.BraceWrapping.AfterStruct = true; | 
|  | Style.BraceWrapping.AfterExternBlock = true; | 
|  | Style.BraceWrapping.BeforeCatch = true; | 
|  | Style.BraceWrapping.BeforeElse = true; | 
|  | Style.PenaltyReturnTypeOnItsOwnLine = 1000; | 
|  | Style.AllowShortFunctionsOnASingleLine = FormatStyle::SFS_None; | 
|  | Style.AllowShortCaseLabelsOnASingleLine = false; | 
|  | Style.AllowShortIfStatementsOnASingleLine = FormatStyle::SIS_Never; | 
|  | Style.AllowShortLoopsOnASingleLine = false; | 
|  | Style.AlwaysBreakAfterDefinitionReturnType = FormatStyle::DRTBS_None; | 
|  | Style.AlwaysBreakAfterReturnType = FormatStyle::RTBS_None; | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | FormatStyle getNoStyle() { | 
|  | FormatStyle NoStyle = getLLVMStyle(); | 
|  | NoStyle.DisableFormat = true; | 
|  | NoStyle.SortIncludes = false; | 
|  | NoStyle.SortUsingDeclarations = false; | 
|  | return NoStyle; | 
|  | } | 
|  |  | 
|  | bool getPredefinedStyle(StringRef Name, FormatStyle::LanguageKind Language, | 
|  | FormatStyle *Style) { | 
|  | if (Name.equals_lower("llvm")) { | 
|  | *Style = getLLVMStyle(Language); | 
|  | } else if (Name.equals_lower("chromium")) { | 
|  | *Style = getChromiumStyle(Language); | 
|  | } else if (Name.equals_lower("mozilla")) { | 
|  | *Style = getMozillaStyle(); | 
|  | } else if (Name.equals_lower("google")) { | 
|  | *Style = getGoogleStyle(Language); | 
|  | } else if (Name.equals_lower("webkit")) { | 
|  | *Style = getWebKitStyle(); | 
|  | } else if (Name.equals_lower("gnu")) { | 
|  | *Style = getGNUStyle(); | 
|  | } else if (Name.equals_lower("microsoft")) { | 
|  | *Style = getMicrosoftStyle(Language); | 
|  | } else if (Name.equals_lower("none")) { | 
|  | *Style = getNoStyle(); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | Style->Language = Language; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | std::error_code parseConfiguration(StringRef Text, FormatStyle *Style) { | 
|  | assert(Style); | 
|  | FormatStyle::LanguageKind Language = Style->Language; | 
|  | assert(Language != FormatStyle::LK_None); | 
|  | if (Text.trim().empty()) | 
|  | return make_error_code(ParseError::Error); | 
|  | Style->StyleSet.Clear(); | 
|  | std::vector<FormatStyle> Styles; | 
|  | llvm::yaml::Input Input(Text); | 
|  | // DocumentListTraits<vector<FormatStyle>> uses the context to get default | 
|  | // values for the fields, keys for which are missing from the configuration. | 
|  | // Mapping also uses the context to get the language to find the correct | 
|  | // base style. | 
|  | Input.setContext(Style); | 
|  | Input >> Styles; | 
|  | if (Input.error()) | 
|  | return Input.error(); | 
|  |  | 
|  | for (unsigned i = 0; i < Styles.size(); ++i) { | 
|  | // Ensures that only the first configuration can skip the Language option. | 
|  | if (Styles[i].Language == FormatStyle::LK_None && i != 0) | 
|  | return make_error_code(ParseError::Error); | 
|  | // Ensure that each language is configured at most once. | 
|  | for (unsigned j = 0; j < i; ++j) { | 
|  | if (Styles[i].Language == Styles[j].Language) { | 
|  | LLVM_DEBUG(llvm::dbgs() | 
|  | << "Duplicate languages in the config file on positions " | 
|  | << j << " and " << i << "\n"); | 
|  | return make_error_code(ParseError::Error); | 
|  | } | 
|  | } | 
|  | } | 
|  | // Look for a suitable configuration starting from the end, so we can | 
|  | // find the configuration for the specific language first, and the default | 
|  | // configuration (which can only be at slot 0) after it. | 
|  | FormatStyle::FormatStyleSet StyleSet; | 
|  | bool LanguageFound = false; | 
|  | for (int i = Styles.size() - 1; i >= 0; --i) { | 
|  | if (Styles[i].Language != FormatStyle::LK_None) | 
|  | StyleSet.Add(Styles[i]); | 
|  | if (Styles[i].Language == Language) | 
|  | LanguageFound = true; | 
|  | } | 
|  | if (!LanguageFound) { | 
|  | if (Styles.empty() || Styles[0].Language != FormatStyle::LK_None) | 
|  | return make_error_code(ParseError::Unsuitable); | 
|  | FormatStyle DefaultStyle = Styles[0]; | 
|  | DefaultStyle.Language = Language; | 
|  | StyleSet.Add(std::move(DefaultStyle)); | 
|  | } | 
|  | *Style = *StyleSet.Get(Language); | 
|  | return make_error_code(ParseError::Success); | 
|  | } | 
|  |  | 
|  | std::string configurationAsText(const FormatStyle &Style) { | 
|  | std::string Text; | 
|  | llvm::raw_string_ostream Stream(Text); | 
|  | llvm::yaml::Output Output(Stream); | 
|  | // We use the same mapping method for input and output, so we need a non-const | 
|  | // reference here. | 
|  | FormatStyle NonConstStyle = expandPresets(Style); | 
|  | Output << NonConstStyle; | 
|  | return Stream.str(); | 
|  | } | 
|  |  | 
|  | llvm::Optional<FormatStyle> | 
|  | FormatStyle::FormatStyleSet::Get(FormatStyle::LanguageKind Language) const { | 
|  | if (!Styles) | 
|  | return None; | 
|  | auto It = Styles->find(Language); | 
|  | if (It == Styles->end()) | 
|  | return None; | 
|  | FormatStyle Style = It->second; | 
|  | Style.StyleSet = *this; | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | void FormatStyle::FormatStyleSet::Add(FormatStyle Style) { | 
|  | assert(Style.Language != LK_None && | 
|  | "Cannot add a style for LK_None to a StyleSet"); | 
|  | assert( | 
|  | !Style.StyleSet.Styles && | 
|  | "Cannot add a style associated with an existing StyleSet to a StyleSet"); | 
|  | if (!Styles) | 
|  | Styles = std::make_shared<MapType>(); | 
|  | (*Styles)[Style.Language] = std::move(Style); | 
|  | } | 
|  |  | 
|  | void FormatStyle::FormatStyleSet::Clear() { Styles.reset(); } | 
|  |  | 
|  | llvm::Optional<FormatStyle> | 
|  | FormatStyle::GetLanguageStyle(FormatStyle::LanguageKind Language) const { | 
|  | return StyleSet.Get(Language); | 
|  | } | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | class JavaScriptRequoter : public TokenAnalyzer { | 
|  | public: | 
|  | JavaScriptRequoter(const Environment &Env, const FormatStyle &Style) | 
|  | : TokenAnalyzer(Env, Style) {} | 
|  |  | 
|  | std::pair<tooling::Replacements, unsigned> | 
|  | analyze(TokenAnnotator &Annotator, | 
|  | SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | FormatTokenLexer &Tokens) override { | 
|  | AffectedRangeMgr.computeAffectedLines(AnnotatedLines); | 
|  | tooling::Replacements Result; | 
|  | requoteJSStringLiteral(AnnotatedLines, Result); | 
|  | return {Result, 0}; | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Replaces double/single-quoted string literal as appropriate, re-escaping | 
|  | // the contents in the process. | 
|  | void requoteJSStringLiteral(SmallVectorImpl<AnnotatedLine *> &Lines, | 
|  | tooling::Replacements &Result) { | 
|  | for (AnnotatedLine *Line : Lines) { | 
|  | requoteJSStringLiteral(Line->Children, Result); | 
|  | if (!Line->Affected) | 
|  | continue; | 
|  | for (FormatToken *FormatTok = Line->First; FormatTok; | 
|  | FormatTok = FormatTok->Next) { | 
|  | StringRef Input = FormatTok->TokenText; | 
|  | if (FormatTok->Finalized || !FormatTok->isStringLiteral() || | 
|  | // NB: testing for not starting with a double quote to avoid | 
|  | // breaking `template strings`. | 
|  | (Style.JavaScriptQuotes == FormatStyle::JSQS_Single && | 
|  | !Input.startswith("\"")) || | 
|  | (Style.JavaScriptQuotes == FormatStyle::JSQS_Double && | 
|  | !Input.startswith("\'"))) | 
|  | continue; | 
|  |  | 
|  | // Change start and end quote. | 
|  | bool IsSingle = Style.JavaScriptQuotes == FormatStyle::JSQS_Single; | 
|  | SourceLocation Start = FormatTok->Tok.getLocation(); | 
|  | auto Replace = [&](SourceLocation Start, unsigned Length, | 
|  | StringRef ReplacementText) { | 
|  | auto Err = Result.add(tooling::Replacement( | 
|  | Env.getSourceManager(), Start, Length, ReplacementText)); | 
|  | // FIXME: handle error. For now, print error message and skip the | 
|  | // replacement for release version. | 
|  | if (Err) { | 
|  | llvm::errs() << llvm::toString(std::move(Err)) << "\n"; | 
|  | assert(false); | 
|  | } | 
|  | }; | 
|  | Replace(Start, 1, IsSingle ? "'" : "\""); | 
|  | Replace(FormatTok->Tok.getEndLoc().getLocWithOffset(-1), 1, | 
|  | IsSingle ? "'" : "\""); | 
|  |  | 
|  | // Escape internal quotes. | 
|  | bool Escaped = false; | 
|  | for (size_t i = 1; i < Input.size() - 1; i++) { | 
|  | switch (Input[i]) { | 
|  | case '\\': | 
|  | if (!Escaped && i + 1 < Input.size() && | 
|  | ((IsSingle && Input[i + 1] == '"') || | 
|  | (!IsSingle && Input[i + 1] == '\''))) { | 
|  | // Remove this \, it's escaping a " or ' that no longer needs | 
|  | // escaping | 
|  | Replace(Start.getLocWithOffset(i), 1, ""); | 
|  | continue; | 
|  | } | 
|  | Escaped = !Escaped; | 
|  | break; | 
|  | case '\"': | 
|  | case '\'': | 
|  | if (!Escaped && IsSingle == (Input[i] == '\'')) { | 
|  | // Escape the quote. | 
|  | Replace(Start.getLocWithOffset(i), 0, "\\"); | 
|  | } | 
|  | Escaped = false; | 
|  | break; | 
|  | default: | 
|  | Escaped = false; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | class Formatter : public TokenAnalyzer { | 
|  | public: | 
|  | Formatter(const Environment &Env, const FormatStyle &Style, | 
|  | FormattingAttemptStatus *Status) | 
|  | : TokenAnalyzer(Env, Style), Status(Status) {} | 
|  |  | 
|  | std::pair<tooling::Replacements, unsigned> | 
|  | analyze(TokenAnnotator &Annotator, | 
|  | SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | FormatTokenLexer &Tokens) override { | 
|  | tooling::Replacements Result; | 
|  | deriveLocalStyle(AnnotatedLines); | 
|  | AffectedRangeMgr.computeAffectedLines(AnnotatedLines); | 
|  | for (unsigned i = 0, e = AnnotatedLines.size(); i != e; ++i) { | 
|  | Annotator.calculateFormattingInformation(*AnnotatedLines[i]); | 
|  | } | 
|  | Annotator.setCommentLineLevels(AnnotatedLines); | 
|  |  | 
|  | WhitespaceManager Whitespaces( | 
|  | Env.getSourceManager(), Style, | 
|  | inputUsesCRLF(Env.getSourceManager().getBufferData(Env.getFileID()))); | 
|  | ContinuationIndenter Indenter(Style, Tokens.getKeywords(), | 
|  | Env.getSourceManager(), Whitespaces, Encoding, | 
|  | BinPackInconclusiveFunctions); | 
|  | unsigned Penalty = | 
|  | UnwrappedLineFormatter(&Indenter, &Whitespaces, Style, | 
|  | Tokens.getKeywords(), Env.getSourceManager(), | 
|  | Status) | 
|  | .format(AnnotatedLines, /*DryRun=*/false, | 
|  | /*AdditionalIndent=*/0, | 
|  | /*FixBadIndentation=*/false, | 
|  | /*FirstStartColumn=*/Env.getFirstStartColumn(), | 
|  | /*NextStartColumn=*/Env.getNextStartColumn(), | 
|  | /*LastStartColumn=*/Env.getLastStartColumn()); | 
|  | for (const auto &R : Whitespaces.generateReplacements()) | 
|  | if (Result.add(R)) | 
|  | return std::make_pair(Result, 0); | 
|  | return std::make_pair(Result, Penalty); | 
|  | } | 
|  |  | 
|  | private: | 
|  | static bool inputUsesCRLF(StringRef Text) { | 
|  | return Text.count('\r') * 2 > Text.count('\n'); | 
|  | } | 
|  |  | 
|  | bool | 
|  | hasCpp03IncompatibleFormat(const SmallVectorImpl<AnnotatedLine *> &Lines) { | 
|  | for (const AnnotatedLine *Line : Lines) { | 
|  | if (hasCpp03IncompatibleFormat(Line->Children)) | 
|  | return true; | 
|  | for (FormatToken *Tok = Line->First->Next; Tok; Tok = Tok->Next) { | 
|  | if (Tok->WhitespaceRange.getBegin() == Tok->WhitespaceRange.getEnd()) { | 
|  | if (Tok->is(tok::coloncolon) && Tok->Previous->is(TT_TemplateOpener)) | 
|  | return true; | 
|  | if (Tok->is(TT_TemplateCloser) && | 
|  | Tok->Previous->is(TT_TemplateCloser)) | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | int countVariableAlignments(const SmallVectorImpl<AnnotatedLine *> &Lines) { | 
|  | int AlignmentDiff = 0; | 
|  | for (const AnnotatedLine *Line : Lines) { | 
|  | AlignmentDiff += countVariableAlignments(Line->Children); | 
|  | for (FormatToken *Tok = Line->First; Tok && Tok->Next; Tok = Tok->Next) { | 
|  | if (!Tok->is(TT_PointerOrReference)) | 
|  | continue; | 
|  | bool SpaceBefore = | 
|  | Tok->WhitespaceRange.getBegin() != Tok->WhitespaceRange.getEnd(); | 
|  | bool SpaceAfter = Tok->Next->WhitespaceRange.getBegin() != | 
|  | Tok->Next->WhitespaceRange.getEnd(); | 
|  | if (SpaceBefore && !SpaceAfter) | 
|  | ++AlignmentDiff; | 
|  | if (!SpaceBefore && SpaceAfter) | 
|  | --AlignmentDiff; | 
|  | } | 
|  | } | 
|  | return AlignmentDiff; | 
|  | } | 
|  |  | 
|  | void | 
|  | deriveLocalStyle(const SmallVectorImpl<AnnotatedLine *> &AnnotatedLines) { | 
|  | bool HasBinPackedFunction = false; | 
|  | bool HasOnePerLineFunction = false; | 
|  | for (unsigned i = 0, e = AnnotatedLines.size(); i != e; ++i) { | 
|  | if (!AnnotatedLines[i]->First->Next) | 
|  | continue; | 
|  | FormatToken *Tok = AnnotatedLines[i]->First->Next; | 
|  | while (Tok->Next) { | 
|  | if (Tok->PackingKind == PPK_BinPacked) | 
|  | HasBinPackedFunction = true; | 
|  | if (Tok->PackingKind == PPK_OnePerLine) | 
|  | HasOnePerLineFunction = true; | 
|  |  | 
|  | Tok = Tok->Next; | 
|  | } | 
|  | } | 
|  | if (Style.DerivePointerAlignment) | 
|  | Style.PointerAlignment = countVariableAlignments(AnnotatedLines) <= 0 | 
|  | ? FormatStyle::PAS_Left | 
|  | : FormatStyle::PAS_Right; | 
|  | if (Style.Standard == FormatStyle::LS_Auto) | 
|  | Style.Standard = hasCpp03IncompatibleFormat(AnnotatedLines) | 
|  | ? FormatStyle::LS_Latest | 
|  | : FormatStyle::LS_Cpp03; | 
|  | BinPackInconclusiveFunctions = | 
|  | HasBinPackedFunction || !HasOnePerLineFunction; | 
|  | } | 
|  |  | 
|  | bool BinPackInconclusiveFunctions; | 
|  | FormattingAttemptStatus *Status; | 
|  | }; | 
|  |  | 
|  | // This class clean up the erroneous/redundant code around the given ranges in | 
|  | // file. | 
|  | class Cleaner : public TokenAnalyzer { | 
|  | public: | 
|  | Cleaner(const Environment &Env, const FormatStyle &Style) | 
|  | : TokenAnalyzer(Env, Style), | 
|  | DeletedTokens(FormatTokenLess(Env.getSourceManager())) {} | 
|  |  | 
|  | // FIXME: eliminate unused parameters. | 
|  | std::pair<tooling::Replacements, unsigned> | 
|  | analyze(TokenAnnotator &Annotator, | 
|  | SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | FormatTokenLexer &Tokens) override { | 
|  | // FIXME: in the current implementation the granularity of affected range | 
|  | // is an annotated line. However, this is not sufficient. Furthermore, | 
|  | // redundant code introduced by replacements does not necessarily | 
|  | // intercept with ranges of replacements that result in the redundancy. | 
|  | // To determine if some redundant code is actually introduced by | 
|  | // replacements(e.g. deletions), we need to come up with a more | 
|  | // sophisticated way of computing affected ranges. | 
|  | AffectedRangeMgr.computeAffectedLines(AnnotatedLines); | 
|  |  | 
|  | checkEmptyNamespace(AnnotatedLines); | 
|  |  | 
|  | for (auto *Line : AnnotatedLines) | 
|  | cleanupLine(Line); | 
|  |  | 
|  | return {generateFixes(), 0}; | 
|  | } | 
|  |  | 
|  | private: | 
|  | void cleanupLine(AnnotatedLine *Line) { | 
|  | for (auto *Child : Line->Children) { | 
|  | cleanupLine(Child); | 
|  | } | 
|  |  | 
|  | if (Line->Affected) { | 
|  | cleanupRight(Line->First, tok::comma, tok::comma); | 
|  | cleanupRight(Line->First, TT_CtorInitializerColon, tok::comma); | 
|  | cleanupRight(Line->First, tok::l_paren, tok::comma); | 
|  | cleanupLeft(Line->First, tok::comma, tok::r_paren); | 
|  | cleanupLeft(Line->First, TT_CtorInitializerComma, tok::l_brace); | 
|  | cleanupLeft(Line->First, TT_CtorInitializerColon, tok::l_brace); | 
|  | cleanupLeft(Line->First, TT_CtorInitializerColon, tok::equal); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool containsOnlyComments(const AnnotatedLine &Line) { | 
|  | for (FormatToken *Tok = Line.First; Tok != nullptr; Tok = Tok->Next) { | 
|  | if (Tok->isNot(tok::comment)) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Iterate through all lines and remove any empty (nested) namespaces. | 
|  | void checkEmptyNamespace(SmallVectorImpl<AnnotatedLine *> &AnnotatedLines) { | 
|  | std::set<unsigned> DeletedLines; | 
|  | for (unsigned i = 0, e = AnnotatedLines.size(); i != e; ++i) { | 
|  | auto &Line = *AnnotatedLines[i]; | 
|  | if (Line.startsWithNamespace()) { | 
|  | checkEmptyNamespace(AnnotatedLines, i, i, DeletedLines); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (auto Line : DeletedLines) { | 
|  | FormatToken *Tok = AnnotatedLines[Line]->First; | 
|  | while (Tok) { | 
|  | deleteToken(Tok); | 
|  | Tok = Tok->Next; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // The function checks if the namespace, which starts from \p CurrentLine, and | 
|  | // its nested namespaces are empty and delete them if they are empty. It also | 
|  | // sets \p NewLine to the last line checked. | 
|  | // Returns true if the current namespace is empty. | 
|  | bool checkEmptyNamespace(SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | unsigned CurrentLine, unsigned &NewLine, | 
|  | std::set<unsigned> &DeletedLines) { | 
|  | unsigned InitLine = CurrentLine, End = AnnotatedLines.size(); | 
|  | if (Style.BraceWrapping.AfterNamespace) { | 
|  | // If the left brace is in a new line, we should consume it first so that | 
|  | // it does not make the namespace non-empty. | 
|  | // FIXME: error handling if there is no left brace. | 
|  | if (!AnnotatedLines[++CurrentLine]->startsWith(tok::l_brace)) { | 
|  | NewLine = CurrentLine; | 
|  | return false; | 
|  | } | 
|  | } else if (!AnnotatedLines[CurrentLine]->endsWith(tok::l_brace)) { | 
|  | return false; | 
|  | } | 
|  | while (++CurrentLine < End) { | 
|  | if (AnnotatedLines[CurrentLine]->startsWith(tok::r_brace)) | 
|  | break; | 
|  |  | 
|  | if (AnnotatedLines[CurrentLine]->startsWithNamespace()) { | 
|  | if (!checkEmptyNamespace(AnnotatedLines, CurrentLine, NewLine, | 
|  | DeletedLines)) | 
|  | return false; | 
|  | CurrentLine = NewLine; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (containsOnlyComments(*AnnotatedLines[CurrentLine])) | 
|  | continue; | 
|  |  | 
|  | // If there is anything other than comments or nested namespaces in the | 
|  | // current namespace, the namespace cannot be empty. | 
|  | NewLine = CurrentLine; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | NewLine = CurrentLine; | 
|  | if (CurrentLine >= End) | 
|  | return false; | 
|  |  | 
|  | // Check if the empty namespace is actually affected by changed ranges. | 
|  | if (!AffectedRangeMgr.affectsCharSourceRange(CharSourceRange::getCharRange( | 
|  | AnnotatedLines[InitLine]->First->Tok.getLocation(), | 
|  | AnnotatedLines[CurrentLine]->Last->Tok.getEndLoc()))) | 
|  | return false; | 
|  |  | 
|  | for (unsigned i = InitLine; i <= CurrentLine; ++i) { | 
|  | DeletedLines.insert(i); | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Checks pairs {start, start->next},..., {end->previous, end} and deletes one | 
|  | // of the token in the pair if the left token has \p LK token kind and the | 
|  | // right token has \p RK token kind. If \p DeleteLeft is true, the left token | 
|  | // is deleted on match; otherwise, the right token is deleted. | 
|  | template <typename LeftKind, typename RightKind> | 
|  | void cleanupPair(FormatToken *Start, LeftKind LK, RightKind RK, | 
|  | bool DeleteLeft) { | 
|  | auto NextNotDeleted = [this](const FormatToken &Tok) -> FormatToken * { | 
|  | for (auto *Res = Tok.Next; Res; Res = Res->Next) | 
|  | if (!Res->is(tok::comment) && | 
|  | DeletedTokens.find(Res) == DeletedTokens.end()) | 
|  | return Res; | 
|  | return nullptr; | 
|  | }; | 
|  | for (auto *Left = Start; Left;) { | 
|  | auto *Right = NextNotDeleted(*Left); | 
|  | if (!Right) | 
|  | break; | 
|  | if (Left->is(LK) && Right->is(RK)) { | 
|  | deleteToken(DeleteLeft ? Left : Right); | 
|  | for (auto *Tok = Left->Next; Tok && Tok != Right; Tok = Tok->Next) | 
|  | deleteToken(Tok); | 
|  | // If the right token is deleted, we should keep the left token | 
|  | // unchanged and pair it with the new right token. | 
|  | if (!DeleteLeft) | 
|  | continue; | 
|  | } | 
|  | Left = Right; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename LeftKind, typename RightKind> | 
|  | void cleanupLeft(FormatToken *Start, LeftKind LK, RightKind RK) { | 
|  | cleanupPair(Start, LK, RK, /*DeleteLeft=*/true); | 
|  | } | 
|  |  | 
|  | template <typename LeftKind, typename RightKind> | 
|  | void cleanupRight(FormatToken *Start, LeftKind LK, RightKind RK) { | 
|  | cleanupPair(Start, LK, RK, /*DeleteLeft=*/false); | 
|  | } | 
|  |  | 
|  | // Delete the given token. | 
|  | inline void deleteToken(FormatToken *Tok) { | 
|  | if (Tok) | 
|  | DeletedTokens.insert(Tok); | 
|  | } | 
|  |  | 
|  | tooling::Replacements generateFixes() { | 
|  | tooling::Replacements Fixes; | 
|  | std::vector<FormatToken *> Tokens; | 
|  | std::copy(DeletedTokens.begin(), DeletedTokens.end(), | 
|  | std::back_inserter(Tokens)); | 
|  |  | 
|  | // Merge multiple continuous token deletions into one big deletion so that | 
|  | // the number of replacements can be reduced. This makes computing affected | 
|  | // ranges more efficient when we run reformat on the changed code. | 
|  | unsigned Idx = 0; | 
|  | while (Idx < Tokens.size()) { | 
|  | unsigned St = Idx, End = Idx; | 
|  | while ((End + 1) < Tokens.size() && | 
|  | Tokens[End]->Next == Tokens[End + 1]) { | 
|  | End++; | 
|  | } | 
|  | auto SR = CharSourceRange::getCharRange(Tokens[St]->Tok.getLocation(), | 
|  | Tokens[End]->Tok.getEndLoc()); | 
|  | auto Err = | 
|  | Fixes.add(tooling::Replacement(Env.getSourceManager(), SR, "")); | 
|  | // FIXME: better error handling. for now just print error message and skip | 
|  | // for the release version. | 
|  | if (Err) { | 
|  | llvm::errs() << llvm::toString(std::move(Err)) << "\n"; | 
|  | assert(false && "Fixes must not conflict!"); | 
|  | } | 
|  | Idx = End + 1; | 
|  | } | 
|  |  | 
|  | return Fixes; | 
|  | } | 
|  |  | 
|  | // Class for less-than inequality comparason for the set `RedundantTokens`. | 
|  | // We store tokens in the order they appear in the translation unit so that | 
|  | // we do not need to sort them in `generateFixes()`. | 
|  | struct FormatTokenLess { | 
|  | FormatTokenLess(const SourceManager &SM) : SM(SM) {} | 
|  |  | 
|  | bool operator()(const FormatToken *LHS, const FormatToken *RHS) const { | 
|  | return SM.isBeforeInTranslationUnit(LHS->Tok.getLocation(), | 
|  | RHS->Tok.getLocation()); | 
|  | } | 
|  | const SourceManager &SM; | 
|  | }; | 
|  |  | 
|  | // Tokens to be deleted. | 
|  | std::set<FormatToken *, FormatTokenLess> DeletedTokens; | 
|  | }; | 
|  |  | 
|  | class ObjCHeaderStyleGuesser : public TokenAnalyzer { | 
|  | public: | 
|  | ObjCHeaderStyleGuesser(const Environment &Env, const FormatStyle &Style) | 
|  | : TokenAnalyzer(Env, Style), IsObjC(false) {} | 
|  |  | 
|  | std::pair<tooling::Replacements, unsigned> | 
|  | analyze(TokenAnnotator &Annotator, | 
|  | SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | FormatTokenLexer &Tokens) override { | 
|  | assert(Style.Language == FormatStyle::LK_Cpp); | 
|  | IsObjC = guessIsObjC(Env.getSourceManager(), AnnotatedLines, | 
|  | Tokens.getKeywords()); | 
|  | tooling::Replacements Result; | 
|  | return {Result, 0}; | 
|  | } | 
|  |  | 
|  | bool isObjC() { return IsObjC; } | 
|  |  | 
|  | private: | 
|  | static bool | 
|  | guessIsObjC(const SourceManager &SourceManager, | 
|  | const SmallVectorImpl<AnnotatedLine *> &AnnotatedLines, | 
|  | const AdditionalKeywords &Keywords) { | 
|  | // Keep this array sorted, since we are binary searching over it. | 
|  | static constexpr llvm::StringLiteral FoundationIdentifiers[] = { | 
|  | "CGFloat", | 
|  | "CGPoint", | 
|  | "CGPointMake", | 
|  | "CGPointZero", | 
|  | "CGRect", | 
|  | "CGRectEdge", | 
|  | "CGRectInfinite", | 
|  | "CGRectMake", | 
|  | "CGRectNull", | 
|  | "CGRectZero", | 
|  | "CGSize", | 
|  | "CGSizeMake", | 
|  | "CGVector", | 
|  | "CGVectorMake", | 
|  | "NSAffineTransform", | 
|  | "NSArray", | 
|  | "NSAttributedString", | 
|  | "NSBlockOperation", | 
|  | "NSBundle", | 
|  | "NSCache", | 
|  | "NSCalendar", | 
|  | "NSCharacterSet", | 
|  | "NSCountedSet", | 
|  | "NSData", | 
|  | "NSDataDetector", | 
|  | "NSDecimal", | 
|  | "NSDecimalNumber", | 
|  | "NSDictionary", | 
|  | "NSEdgeInsets", | 
|  | "NSHashTable", | 
|  | "NSIndexPath", | 
|  | "NSIndexSet", | 
|  | "NSInteger", | 
|  | "NSInvocationOperation", | 
|  | "NSLocale", | 
|  | "NSMapTable", | 
|  | "NSMutableArray", | 
|  | "NSMutableAttributedString", | 
|  | "NSMutableCharacterSet", | 
|  | "NSMutableData", | 
|  | "NSMutableDictionary", | 
|  | "NSMutableIndexSet", | 
|  | "NSMutableOrderedSet", | 
|  | "NSMutableSet", | 
|  | "NSMutableString", | 
|  | "NSNumber", | 
|  | "NSNumberFormatter", | 
|  | "NSObject", | 
|  | "NSOperation", | 
|  | "NSOperationQueue", | 
|  | "NSOperationQueuePriority", | 
|  | "NSOrderedSet", | 
|  | "NSPoint", | 
|  | "NSPointerArray", | 
|  | "NSQualityOfService", | 
|  | "NSRange", | 
|  | "NSRect", | 
|  | "NSRegularExpression", | 
|  | "NSSet", | 
|  | "NSSize", | 
|  | "NSString", | 
|  | "NSTimeZone", | 
|  | "NSUInteger", | 
|  | "NSURL", | 
|  | "NSURLComponents", | 
|  | "NSURLQueryItem", | 
|  | "NSUUID", | 
|  | "NSValue", | 
|  | "UIImage", | 
|  | "UIView", | 
|  | }; | 
|  |  | 
|  | for (auto Line : AnnotatedLines) { | 
|  | for (const FormatToken *FormatTok = Line->First; FormatTok; | 
|  | FormatTok = FormatTok->Next) { | 
|  | if ((FormatTok->Previous && FormatTok->Previous->is(tok::at) && | 
|  | (FormatTok->Tok.getObjCKeywordID() != tok::objc_not_keyword || | 
|  | FormatTok->isOneOf(tok::numeric_constant, tok::l_square, | 
|  | tok::l_brace))) || | 
|  | (FormatTok->Tok.isAnyIdentifier() && | 
|  | std::binary_search(std::begin(FoundationIdentifiers), | 
|  | std::end(FoundationIdentifiers), | 
|  | FormatTok->TokenText)) || | 
|  | FormatTok->is(TT_ObjCStringLiteral) || | 
|  | FormatTok->isOneOf(Keywords.kw_NS_CLOSED_ENUM, Keywords.kw_NS_ENUM, | 
|  | Keywords.kw_NS_OPTIONS, TT_ObjCBlockLBrace, | 
|  | TT_ObjCBlockLParen, TT_ObjCDecl, TT_ObjCForIn, | 
|  | TT_ObjCMethodExpr, TT_ObjCMethodSpecifier, | 
|  | TT_ObjCProperty)) { | 
|  | LLVM_DEBUG(llvm::dbgs() | 
|  | << "Detected ObjC at location " | 
|  | << FormatTok->Tok.getLocation().printToString( | 
|  | SourceManager) | 
|  | << " token: " << FormatTok->TokenText << " token type: " | 
|  | << getTokenTypeName(FormatTok->Type) << "\n"); | 
|  | return true; | 
|  | } | 
|  | if (guessIsObjC(SourceManager, Line->Children, Keywords)) | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool IsObjC; | 
|  | }; | 
|  |  | 
|  | struct IncludeDirective { | 
|  | StringRef Filename; | 
|  | StringRef Text; | 
|  | unsigned Offset; | 
|  | int Category; | 
|  | int Priority; | 
|  | }; | 
|  |  | 
|  | struct JavaImportDirective { | 
|  | StringRef Identifier; | 
|  | StringRef Text; | 
|  | unsigned Offset; | 
|  | std::vector<StringRef> AssociatedCommentLines; | 
|  | bool IsStatic; | 
|  | }; | 
|  |  | 
|  | } // end anonymous namespace | 
|  |  | 
|  | // Determines whether 'Ranges' intersects with ('Start', 'End'). | 
|  | static bool affectsRange(ArrayRef<tooling::Range> Ranges, unsigned Start, | 
|  | unsigned End) { | 
|  | for (auto Range : Ranges) { | 
|  | if (Range.getOffset() < End && | 
|  | Range.getOffset() + Range.getLength() > Start) | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Returns a pair (Index, OffsetToEOL) describing the position of the cursor | 
|  | // before sorting/deduplicating. Index is the index of the include under the | 
|  | // cursor in the original set of includes. If this include has duplicates, it is | 
|  | // the index of the first of the duplicates as the others are going to be | 
|  | // removed. OffsetToEOL describes the cursor's position relative to the end of | 
|  | // its current line. | 
|  | // If `Cursor` is not on any #include, `Index` will be UINT_MAX. | 
|  | static std::pair<unsigned, unsigned> | 
|  | FindCursorIndex(const SmallVectorImpl<IncludeDirective> &Includes, | 
|  | const SmallVectorImpl<unsigned> &Indices, unsigned Cursor) { | 
|  | unsigned CursorIndex = UINT_MAX; | 
|  | unsigned OffsetToEOL = 0; | 
|  | for (int i = 0, e = Includes.size(); i != e; ++i) { | 
|  | unsigned Start = Includes[Indices[i]].Offset; | 
|  | unsigned End = Start + Includes[Indices[i]].Text.size(); | 
|  | if (!(Cursor >= Start && Cursor < End)) | 
|  | continue; | 
|  | CursorIndex = Indices[i]; | 
|  | OffsetToEOL = End - Cursor; | 
|  | // Put the cursor on the only remaining #include among the duplicate | 
|  | // #includes. | 
|  | while (--i >= 0 && Includes[CursorIndex].Text == Includes[Indices[i]].Text) | 
|  | CursorIndex = i; | 
|  | break; | 
|  | } | 
|  | return std::make_pair(CursorIndex, OffsetToEOL); | 
|  | } | 
|  |  | 
|  | // Replace all "\r\n" with "\n". | 
|  | std::string replaceCRLF(const std::string &Code) { | 
|  | std::string NewCode; | 
|  | size_t Pos = 0, LastPos = 0; | 
|  |  | 
|  | do { | 
|  | Pos = Code.find("\r\n", LastPos); | 
|  | if (Pos == LastPos) { | 
|  | LastPos++; | 
|  | continue; | 
|  | } | 
|  | if (Pos == std::string::npos) { | 
|  | NewCode += Code.substr(LastPos); | 
|  | break; | 
|  | } | 
|  | NewCode += Code.substr(LastPos, Pos - LastPos) + "\n"; | 
|  | LastPos = Pos + 2; | 
|  | } while (Pos != std::string::npos); | 
|  |  | 
|  | return NewCode; | 
|  | } | 
|  |  | 
|  | // Sorts and deduplicate a block of includes given by 'Includes' alphabetically | 
|  | // adding the necessary replacement to 'Replaces'. 'Includes' must be in strict | 
|  | // source order. | 
|  | // #include directives with the same text will be deduplicated, and only the | 
|  | // first #include in the duplicate #includes remains. If the `Cursor` is | 
|  | // provided and put on a deleted #include, it will be moved to the remaining | 
|  | // #include in the duplicate #includes. | 
|  | static void sortCppIncludes(const FormatStyle &Style, | 
|  | const SmallVectorImpl<IncludeDirective> &Includes, | 
|  | ArrayRef<tooling::Range> Ranges, StringRef FileName, | 
|  | StringRef Code, tooling::Replacements &Replaces, | 
|  | unsigned *Cursor) { | 
|  | tooling::IncludeCategoryManager Categories(Style.IncludeStyle, FileName); | 
|  | unsigned IncludesBeginOffset = Includes.front().Offset; | 
|  | unsigned IncludesEndOffset = | 
|  | Includes.back().Offset + Includes.back().Text.size(); | 
|  | unsigned IncludesBlockSize = IncludesEndOffset - IncludesBeginOffset; | 
|  | if (!affectsRange(Ranges, IncludesBeginOffset, IncludesEndOffset)) | 
|  | return; | 
|  | SmallVector<unsigned, 16> Indices; | 
|  | for (unsigned i = 0, e = Includes.size(); i != e; ++i) { | 
|  | Indices.push_back(i); | 
|  | } | 
|  | llvm::stable_sort(Indices, [&](unsigned LHSI, unsigned RHSI) { | 
|  | return std::tie(Includes[LHSI].Priority, Includes[LHSI].Filename) < | 
|  | std::tie(Includes[RHSI].Priority, Includes[RHSI].Filename); | 
|  | }); | 
|  | // The index of the include on which the cursor will be put after | 
|  | // sorting/deduplicating. | 
|  | unsigned CursorIndex; | 
|  | // The offset from cursor to the end of line. | 
|  | unsigned CursorToEOLOffset; | 
|  | if (Cursor) | 
|  | std::tie(CursorIndex, CursorToEOLOffset) = | 
|  | FindCursorIndex(Includes, Indices, *Cursor); | 
|  |  | 
|  | // Deduplicate #includes. | 
|  | Indices.erase(std::unique(Indices.begin(), Indices.end(), | 
|  | [&](unsigned LHSI, unsigned RHSI) { | 
|  | return Includes[LHSI].Text == Includes[RHSI].Text; | 
|  | }), | 
|  | Indices.end()); | 
|  |  | 
|  | int CurrentCategory = Includes.front().Category; | 
|  |  | 
|  | // If the #includes are out of order, we generate a single replacement fixing | 
|  | // the entire block. Otherwise, no replacement is generated. | 
|  | // In case Style.IncldueStyle.IncludeBlocks != IBS_Preserve, this check is not | 
|  | // enough as additional newlines might be added or removed across #include | 
|  | // blocks. This we handle below by generating the updated #imclude blocks and | 
|  | // comparing it to the original. | 
|  | if (Indices.size() == Includes.size() && | 
|  | std::is_sorted(Indices.begin(), Indices.end()) && | 
|  | Style.IncludeStyle.IncludeBlocks == tooling::IncludeStyle::IBS_Preserve) | 
|  | return; | 
|  |  | 
|  | std::string result; | 
|  | for (unsigned Index : Indices) { | 
|  | if (!result.empty()) { | 
|  | result += "\n"; | 
|  | if (Style.IncludeStyle.IncludeBlocks == | 
|  | tooling::IncludeStyle::IBS_Regroup && | 
|  | CurrentCategory != Includes[Index].Category) | 
|  | result += "\n"; | 
|  | } | 
|  | result += Includes[Index].Text; | 
|  | if (Cursor && CursorIndex == Index) | 
|  | *Cursor = IncludesBeginOffset + result.size() - CursorToEOLOffset; | 
|  | CurrentCategory = Includes[Index].Category; | 
|  | } | 
|  |  | 
|  | // If the #includes are out of order, we generate a single replacement fixing | 
|  | // the entire range of blocks. Otherwise, no replacement is generated. | 
|  | if (replaceCRLF(result) == | 
|  | replaceCRLF(Code.substr(IncludesBeginOffset, IncludesBlockSize))) | 
|  | return; | 
|  |  | 
|  | auto Err = Replaces.add(tooling::Replacement( | 
|  | FileName, Includes.front().Offset, IncludesBlockSize, result)); | 
|  | // FIXME: better error handling. For now, just skip the replacement for the | 
|  | // release version. | 
|  | if (Err) { | 
|  | llvm::errs() << llvm::toString(std::move(Err)) << "\n"; | 
|  | assert(false); | 
|  | } | 
|  | } | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char CppIncludeRegexPattern[] = | 
|  | R"(^[\t\ ]*#[\t\ ]*(import|include)[^"<]*(["<][^">]*[">]))"; | 
|  |  | 
|  | } // anonymous namespace | 
|  |  | 
|  | tooling::Replacements sortCppIncludes(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName, | 
|  | tooling::Replacements &Replaces, | 
|  | unsigned *Cursor) { | 
|  | unsigned Prev = 0; | 
|  | unsigned SearchFrom = 0; | 
|  | llvm::Regex IncludeRegex(CppIncludeRegexPattern); | 
|  | SmallVector<StringRef, 4> Matches; | 
|  | SmallVector<IncludeDirective, 16> IncludesInBlock; | 
|  |  | 
|  | // In compiled files, consider the first #include to be the main #include of | 
|  | // the file if it is not a system #include. This ensures that the header | 
|  | // doesn't have hidden dependencies | 
|  | // (http://llvm.org/docs/CodingStandards.html#include-style). | 
|  | // | 
|  | // FIXME: Do some sanity checking, e.g. edit distance of the base name, to fix | 
|  | // cases where the first #include is unlikely to be the main header. | 
|  | tooling::IncludeCategoryManager Categories(Style.IncludeStyle, FileName); | 
|  | bool FirstIncludeBlock = true; | 
|  | bool MainIncludeFound = false; | 
|  | bool FormattingOff = false; | 
|  |  | 
|  | for (;;) { | 
|  | auto Pos = Code.find('\n', SearchFrom); | 
|  | StringRef Line = | 
|  | Code.substr(Prev, (Pos != StringRef::npos ? Pos : Code.size()) - Prev); | 
|  |  | 
|  | StringRef Trimmed = Line.trim(); | 
|  | if (Trimmed == "// clang-format off" || Trimmed == "/* clang-format off */") | 
|  | FormattingOff = true; | 
|  | else if (Trimmed == "// clang-format on" || | 
|  | Trimmed == "/* clang-format on */") | 
|  | FormattingOff = false; | 
|  |  | 
|  | const bool EmptyLineSkipped = | 
|  | Trimmed.empty() && | 
|  | (Style.IncludeStyle.IncludeBlocks == tooling::IncludeStyle::IBS_Merge || | 
|  | Style.IncludeStyle.IncludeBlocks == | 
|  | tooling::IncludeStyle::IBS_Regroup); | 
|  |  | 
|  | if (!FormattingOff && !Line.endswith("\\")) { | 
|  | if (IncludeRegex.match(Line, &Matches)) { | 
|  | StringRef IncludeName = Matches[2]; | 
|  | int Category = Categories.getIncludePriority( | 
|  | IncludeName, | 
|  | /*CheckMainHeader=*/!MainIncludeFound && FirstIncludeBlock); | 
|  | int Priority = Categories.getSortIncludePriority( | 
|  | IncludeName, !MainIncludeFound && FirstIncludeBlock); | 
|  | if (Category == 0) | 
|  | MainIncludeFound = true; | 
|  | IncludesInBlock.push_back( | 
|  | {IncludeName, Line, Prev, Category, Priority}); | 
|  | } else if (!IncludesInBlock.empty() && !EmptyLineSkipped) { | 
|  | sortCppIncludes(Style, IncludesInBlock, Ranges, FileName, Code, | 
|  | Replaces, Cursor); | 
|  | IncludesInBlock.clear(); | 
|  | FirstIncludeBlock = false; | 
|  | } | 
|  | Prev = Pos + 1; | 
|  | } | 
|  | if (Pos == StringRef::npos || Pos + 1 == Code.size()) | 
|  | break; | 
|  | SearchFrom = Pos + 1; | 
|  | } | 
|  | if (!IncludesInBlock.empty()) { | 
|  | sortCppIncludes(Style, IncludesInBlock, Ranges, FileName, Code, Replaces, | 
|  | Cursor); | 
|  | } | 
|  | return Replaces; | 
|  | } | 
|  |  | 
|  | // Returns group number to use as a first order sort on imports. Gives UINT_MAX | 
|  | // if the import does not match any given groups. | 
|  | static unsigned findJavaImportGroup(const FormatStyle &Style, | 
|  | StringRef ImportIdentifier) { | 
|  | unsigned LongestMatchIndex = UINT_MAX; | 
|  | unsigned LongestMatchLength = 0; | 
|  | for (unsigned I = 0; I < Style.JavaImportGroups.size(); I++) { | 
|  | std::string GroupPrefix = Style.JavaImportGroups[I]; | 
|  | if (ImportIdentifier.startswith(GroupPrefix) && | 
|  | GroupPrefix.length() > LongestMatchLength) { | 
|  | LongestMatchIndex = I; | 
|  | LongestMatchLength = GroupPrefix.length(); | 
|  | } | 
|  | } | 
|  | return LongestMatchIndex; | 
|  | } | 
|  |  | 
|  | // Sorts and deduplicates a block of includes given by 'Imports' based on | 
|  | // JavaImportGroups, then adding the necessary replacement to 'Replaces'. | 
|  | // Import declarations with the same text will be deduplicated. Between each | 
|  | // import group, a newline is inserted, and within each import group, a | 
|  | // lexicographic sort based on ASCII value is performed. | 
|  | static void sortJavaImports(const FormatStyle &Style, | 
|  | const SmallVectorImpl<JavaImportDirective> &Imports, | 
|  | ArrayRef<tooling::Range> Ranges, StringRef FileName, | 
|  | StringRef Code, tooling::Replacements &Replaces) { | 
|  | unsigned ImportsBeginOffset = Imports.front().Offset; | 
|  | unsigned ImportsEndOffset = | 
|  | Imports.back().Offset + Imports.back().Text.size(); | 
|  | unsigned ImportsBlockSize = ImportsEndOffset - ImportsBeginOffset; | 
|  | if (!affectsRange(Ranges, ImportsBeginOffset, ImportsEndOffset)) | 
|  | return; | 
|  | SmallVector<unsigned, 16> Indices; | 
|  | SmallVector<unsigned, 16> JavaImportGroups; | 
|  | for (unsigned i = 0, e = Imports.size(); i != e; ++i) { | 
|  | Indices.push_back(i); | 
|  | JavaImportGroups.push_back( | 
|  | findJavaImportGroup(Style, Imports[i].Identifier)); | 
|  | } | 
|  | llvm::sort(Indices, [&](unsigned LHSI, unsigned RHSI) { | 
|  | // Negating IsStatic to push static imports above non-static imports. | 
|  | return std::make_tuple(!Imports[LHSI].IsStatic, JavaImportGroups[LHSI], | 
|  | Imports[LHSI].Identifier) < | 
|  | std::make_tuple(!Imports[RHSI].IsStatic, JavaImportGroups[RHSI], | 
|  | Imports[RHSI].Identifier); | 
|  | }); | 
|  |  | 
|  | // Deduplicate imports. | 
|  | Indices.erase(std::unique(Indices.begin(), Indices.end(), | 
|  | [&](unsigned LHSI, unsigned RHSI) { | 
|  | return Imports[LHSI].Text == Imports[RHSI].Text; | 
|  | }), | 
|  | Indices.end()); | 
|  |  | 
|  | bool CurrentIsStatic = Imports[Indices.front()].IsStatic; | 
|  | unsigned CurrentImportGroup = JavaImportGroups[Indices.front()]; | 
|  |  | 
|  | std::string result; | 
|  | for (unsigned Index : Indices) { | 
|  | if (!result.empty()) { | 
|  | result += "\n"; | 
|  | if (CurrentIsStatic != Imports[Index].IsStatic || | 
|  | CurrentImportGroup != JavaImportGroups[Index]) | 
|  | result += "\n"; | 
|  | } | 
|  | for (StringRef CommentLine : Imports[Index].AssociatedCommentLines) { | 
|  | result += CommentLine; | 
|  | result += "\n"; | 
|  | } | 
|  | result += Imports[Index].Text; | 
|  | CurrentIsStatic = Imports[Index].IsStatic; | 
|  | CurrentImportGroup = JavaImportGroups[Index]; | 
|  | } | 
|  |  | 
|  | // If the imports are out of order, we generate a single replacement fixing | 
|  | // the entire block. Otherwise, no replacement is generated. | 
|  | if (replaceCRLF(result) == | 
|  | replaceCRLF(Code.substr(Imports.front().Offset, ImportsBlockSize))) | 
|  | return; | 
|  |  | 
|  | auto Err = Replaces.add(tooling::Replacement(FileName, Imports.front().Offset, | 
|  | ImportsBlockSize, result)); | 
|  | // FIXME: better error handling. For now, just skip the replacement for the | 
|  | // release version. | 
|  | if (Err) { | 
|  | llvm::errs() << llvm::toString(std::move(Err)) << "\n"; | 
|  | assert(false); | 
|  | } | 
|  | } | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char JavaImportRegexPattern[] = | 
|  | "^[\t ]*import[\t ]+(static[\t ]*)?([^\t ]*)[\t ]*;"; | 
|  |  | 
|  | } // anonymous namespace | 
|  |  | 
|  | tooling::Replacements sortJavaImports(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName, | 
|  | tooling::Replacements &Replaces) { | 
|  | unsigned Prev = 0; | 
|  | unsigned SearchFrom = 0; | 
|  | llvm::Regex ImportRegex(JavaImportRegexPattern); | 
|  | SmallVector<StringRef, 4> Matches; | 
|  | SmallVector<JavaImportDirective, 16> ImportsInBlock; | 
|  | std::vector<StringRef> AssociatedCommentLines; | 
|  |  | 
|  | bool FormattingOff = false; | 
|  |  | 
|  | for (;;) { | 
|  | auto Pos = Code.find('\n', SearchFrom); | 
|  | StringRef Line = | 
|  | Code.substr(Prev, (Pos != StringRef::npos ? Pos : Code.size()) - Prev); | 
|  |  | 
|  | StringRef Trimmed = Line.trim(); | 
|  | if (Trimmed == "// clang-format off") | 
|  | FormattingOff = true; | 
|  | else if (Trimmed == "// clang-format on") | 
|  | FormattingOff = false; | 
|  |  | 
|  | if (ImportRegex.match(Line, &Matches)) { | 
|  | if (FormattingOff) { | 
|  | // If at least one import line has formatting turned off, turn off | 
|  | // formatting entirely. | 
|  | return Replaces; | 
|  | } | 
|  | StringRef Static = Matches[1]; | 
|  | StringRef Identifier = Matches[2]; | 
|  | bool IsStatic = false; | 
|  | if (Static.contains("static")) { | 
|  | IsStatic = true; | 
|  | } | 
|  | ImportsInBlock.push_back( | 
|  | {Identifier, Line, Prev, AssociatedCommentLines, IsStatic}); | 
|  | AssociatedCommentLines.clear(); | 
|  | } else if (Trimmed.size() > 0 && !ImportsInBlock.empty()) { | 
|  | // Associating comments within the imports with the nearest import below | 
|  | AssociatedCommentLines.push_back(Line); | 
|  | } | 
|  | Prev = Pos + 1; | 
|  | if (Pos == StringRef::npos || Pos + 1 == Code.size()) | 
|  | break; | 
|  | SearchFrom = Pos + 1; | 
|  | } | 
|  | if (!ImportsInBlock.empty()) | 
|  | sortJavaImports(Style, ImportsInBlock, Ranges, FileName, Code, Replaces); | 
|  | return Replaces; | 
|  | } | 
|  |  | 
|  | bool isMpegTS(StringRef Code) { | 
|  | // MPEG transport streams use the ".ts" file extension. clang-format should | 
|  | // not attempt to format those. MPEG TS' frame format starts with 0x47 every | 
|  | // 189 bytes - detect that and return. | 
|  | return Code.size() > 188 && Code[0] == 0x47 && Code[188] == 0x47; | 
|  | } | 
|  |  | 
|  | bool isLikelyXml(StringRef Code) { return Code.ltrim().startswith("<"); } | 
|  |  | 
|  | tooling::Replacements sortIncludes(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName, unsigned *Cursor) { | 
|  | tooling::Replacements Replaces; | 
|  | if (!Style.SortIncludes) | 
|  | return Replaces; | 
|  | if (isLikelyXml(Code)) | 
|  | return Replaces; | 
|  | if (Style.Language == FormatStyle::LanguageKind::LK_JavaScript && | 
|  | isMpegTS(Code)) | 
|  | return Replaces; | 
|  | if (Style.Language == FormatStyle::LanguageKind::LK_JavaScript) | 
|  | return sortJavaScriptImports(Style, Code, Ranges, FileName); | 
|  | if (Style.Language == FormatStyle::LanguageKind::LK_Java) | 
|  | return sortJavaImports(Style, Code, Ranges, FileName, Replaces); | 
|  | sortCppIncludes(Style, Code, Ranges, FileName, Replaces, Cursor); | 
|  | return Replaces; | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | static llvm::Expected<tooling::Replacements> | 
|  | processReplacements(T ProcessFunc, StringRef Code, | 
|  | const tooling::Replacements &Replaces, | 
|  | const FormatStyle &Style) { | 
|  | if (Replaces.empty()) | 
|  | return tooling::Replacements(); | 
|  |  | 
|  | auto NewCode = applyAllReplacements(Code, Replaces); | 
|  | if (!NewCode) | 
|  | return NewCode.takeError(); | 
|  | std::vector<tooling::Range> ChangedRanges = Replaces.getAffectedRanges(); | 
|  | StringRef FileName = Replaces.begin()->getFilePath(); | 
|  |  | 
|  | tooling::Replacements FormatReplaces = | 
|  | ProcessFunc(Style, *NewCode, ChangedRanges, FileName); | 
|  |  | 
|  | return Replaces.merge(FormatReplaces); | 
|  | } | 
|  |  | 
|  | llvm::Expected<tooling::Replacements> | 
|  | formatReplacements(StringRef Code, const tooling::Replacements &Replaces, | 
|  | const FormatStyle &Style) { | 
|  | // We need to use lambda function here since there are two versions of | 
|  | // `sortIncludes`. | 
|  | auto SortIncludes = [](const FormatStyle &Style, StringRef Code, | 
|  | std::vector<tooling::Range> Ranges, | 
|  | StringRef FileName) -> tooling::Replacements { | 
|  | return sortIncludes(Style, Code, Ranges, FileName); | 
|  | }; | 
|  | auto SortedReplaces = | 
|  | processReplacements(SortIncludes, Code, Replaces, Style); | 
|  | if (!SortedReplaces) | 
|  | return SortedReplaces.takeError(); | 
|  |  | 
|  | // We need to use lambda function here since there are two versions of | 
|  | // `reformat`. | 
|  | auto Reformat = [](const FormatStyle &Style, StringRef Code, | 
|  | std::vector<tooling::Range> Ranges, | 
|  | StringRef FileName) -> tooling::Replacements { | 
|  | return reformat(Style, Code, Ranges, FileName); | 
|  | }; | 
|  | return processReplacements(Reformat, Code, *SortedReplaces, Style); | 
|  | } | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | inline bool isHeaderInsertion(const tooling::Replacement &Replace) { | 
|  | return Replace.getOffset() == UINT_MAX && Replace.getLength() == 0 && | 
|  | llvm::Regex(CppIncludeRegexPattern) | 
|  | .match(Replace.getReplacementText()); | 
|  | } | 
|  |  | 
|  | inline bool isHeaderDeletion(const tooling::Replacement &Replace) { | 
|  | return Replace.getOffset() == UINT_MAX && Replace.getLength() == 1; | 
|  | } | 
|  |  | 
|  | // FIXME: insert empty lines between newly created blocks. | 
|  | tooling::Replacements | 
|  | fixCppIncludeInsertions(StringRef Code, const tooling::Replacements &Replaces, | 
|  | const FormatStyle &Style) { | 
|  | if (!Style.isCpp()) | 
|  | return Replaces; | 
|  |  | 
|  | tooling::Replacements HeaderInsertions; | 
|  | std::set<llvm::StringRef> HeadersToDelete; | 
|  | tooling::Replacements Result; | 
|  | for (const auto &R : Replaces) { | 
|  | if (isHeaderInsertion(R)) { | 
|  | // Replacements from \p Replaces must be conflict-free already, so we can | 
|  | // simply consume the error. | 
|  | llvm::consumeError(HeaderInsertions.add(R)); | 
|  | } else if (isHeaderDeletion(R)) { | 
|  | HeadersToDelete.insert(R.getReplacementText()); | 
|  | } else if (R.getOffset() == UINT_MAX) { | 
|  | llvm::errs() << "Insertions other than header #include insertion are " | 
|  | "not supported! " | 
|  | << R.getReplacementText() << "\n"; | 
|  | } else { | 
|  | llvm::consumeError(Result.add(R)); | 
|  | } | 
|  | } | 
|  | if (HeaderInsertions.empty() && HeadersToDelete.empty()) | 
|  | return Replaces; | 
|  |  | 
|  | StringRef FileName = Replaces.begin()->getFilePath(); | 
|  | tooling::HeaderIncludes Includes(FileName, Code, Style.IncludeStyle); | 
|  |  | 
|  | for (const auto &Header : HeadersToDelete) { | 
|  | tooling::Replacements Replaces = | 
|  | Includes.remove(Header.trim("\"<>"), Header.startswith("<")); | 
|  | for (const auto &R : Replaces) { | 
|  | auto Err = Result.add(R); | 
|  | if (Err) { | 
|  | // Ignore the deletion on conflict. | 
|  | llvm::errs() << "Failed to add header deletion replacement for " | 
|  | << Header << ": " << llvm::toString(std::move(Err)) | 
|  | << "\n"; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | llvm::Regex IncludeRegex = llvm::Regex(CppIncludeRegexPattern); | 
|  | llvm::SmallVector<StringRef, 4> Matches; | 
|  | for (const auto &R : HeaderInsertions) { | 
|  | auto IncludeDirective = R.getReplacementText(); | 
|  | bool Matched = IncludeRegex.match(IncludeDirective, &Matches); | 
|  | assert(Matched && "Header insertion replacement must have replacement text " | 
|  | "'#include ...'"); | 
|  | (void)Matched; | 
|  | auto IncludeName = Matches[2]; | 
|  | auto Replace = | 
|  | Includes.insert(IncludeName.trim("\"<>"), IncludeName.startswith("<")); | 
|  | if (Replace) { | 
|  | auto Err = Result.add(*Replace); | 
|  | if (Err) { | 
|  | llvm::consumeError(std::move(Err)); | 
|  | unsigned NewOffset = | 
|  | Result.getShiftedCodePosition(Replace->getOffset()); | 
|  | auto Shifted = tooling::Replacement(FileName, NewOffset, 0, | 
|  | Replace->getReplacementText()); | 
|  | Result = Result.merge(tooling::Replacements(Shifted)); | 
|  | } | 
|  | } | 
|  | } | 
|  | return Result; | 
|  | } | 
|  |  | 
|  | } // anonymous namespace | 
|  |  | 
|  | llvm::Expected<tooling::Replacements> | 
|  | cleanupAroundReplacements(StringRef Code, const tooling::Replacements &Replaces, | 
|  | const FormatStyle &Style) { | 
|  | // We need to use lambda function here since there are two versions of | 
|  | // `cleanup`. | 
|  | auto Cleanup = [](const FormatStyle &Style, StringRef Code, | 
|  | std::vector<tooling::Range> Ranges, | 
|  | StringRef FileName) -> tooling::Replacements { | 
|  | return cleanup(Style, Code, Ranges, FileName); | 
|  | }; | 
|  | // Make header insertion replacements insert new headers into correct blocks. | 
|  | tooling::Replacements NewReplaces = | 
|  | fixCppIncludeInsertions(Code, Replaces, Style); | 
|  | return processReplacements(Cleanup, Code, NewReplaces, Style); | 
|  | } | 
|  |  | 
|  | namespace internal { | 
|  | std::pair<tooling::Replacements, unsigned> | 
|  | reformat(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, unsigned FirstStartColumn, | 
|  | unsigned NextStartColumn, unsigned LastStartColumn, StringRef FileName, | 
|  | FormattingAttemptStatus *Status) { | 
|  | FormatStyle Expanded = expandPresets(Style); | 
|  | if (Expanded.DisableFormat) | 
|  | return {tooling::Replacements(), 0}; | 
|  | if (isLikelyXml(Code)) | 
|  | return {tooling::Replacements(), 0}; | 
|  | if (Expanded.Language == FormatStyle::LK_JavaScript && isMpegTS(Code)) | 
|  | return {tooling::Replacements(), 0}; | 
|  |  | 
|  | typedef std::function<std::pair<tooling::Replacements, unsigned>( | 
|  | const Environment &)> | 
|  | AnalyzerPass; | 
|  | SmallVector<AnalyzerPass, 4> Passes; | 
|  |  | 
|  | if (Style.Language == FormatStyle::LK_Cpp) { | 
|  | if (Style.FixNamespaceComments) | 
|  | Passes.emplace_back([&](const Environment &Env) { | 
|  | return NamespaceEndCommentsFixer(Env, Expanded).process(); | 
|  | }); | 
|  |  | 
|  | if (Style.SortUsingDeclarations) | 
|  | Passes.emplace_back([&](const Environment &Env) { | 
|  | return UsingDeclarationsSorter(Env, Expanded).process(); | 
|  | }); | 
|  | } | 
|  |  | 
|  | if (Style.Language == FormatStyle::LK_JavaScript && | 
|  | Style.JavaScriptQuotes != FormatStyle::JSQS_Leave) | 
|  | Passes.emplace_back([&](const Environment &Env) { | 
|  | return JavaScriptRequoter(Env, Expanded).process(); | 
|  | }); | 
|  |  | 
|  | Passes.emplace_back([&](const Environment &Env) { | 
|  | return Formatter(Env, Expanded, Status).process(); | 
|  | }); | 
|  |  | 
|  | auto Env = | 
|  | std::make_unique<Environment>(Code, FileName, Ranges, FirstStartColumn, | 
|  | NextStartColumn, LastStartColumn); | 
|  | llvm::Optional<std::string> CurrentCode = None; | 
|  | tooling::Replacements Fixes; | 
|  | unsigned Penalty = 0; | 
|  | for (size_t I = 0, E = Passes.size(); I < E; ++I) { | 
|  | std::pair<tooling::Replacements, unsigned> PassFixes = Passes[I](*Env); | 
|  | auto NewCode = applyAllReplacements( | 
|  | CurrentCode ? StringRef(*CurrentCode) : Code, PassFixes.first); | 
|  | if (NewCode) { | 
|  | Fixes = Fixes.merge(PassFixes.first); | 
|  | Penalty += PassFixes.second; | 
|  | if (I + 1 < E) { | 
|  | CurrentCode = std::move(*NewCode); | 
|  | Env = std::make_unique<Environment>( | 
|  | *CurrentCode, FileName, | 
|  | tooling::calculateRangesAfterReplacements(Fixes, Ranges), | 
|  | FirstStartColumn, NextStartColumn, LastStartColumn); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return {Fixes, Penalty}; | 
|  | } | 
|  | } // namespace internal | 
|  |  | 
|  | tooling::Replacements reformat(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName, | 
|  | FormattingAttemptStatus *Status) { | 
|  | return internal::reformat(Style, Code, Ranges, | 
|  | /*FirstStartColumn=*/0, | 
|  | /*NextStartColumn=*/0, | 
|  | /*LastStartColumn=*/0, FileName, Status) | 
|  | .first; | 
|  | } | 
|  |  | 
|  | tooling::Replacements cleanup(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName) { | 
|  | // cleanups only apply to C++ (they mostly concern ctor commas etc.) | 
|  | if (Style.Language != FormatStyle::LK_Cpp) | 
|  | return tooling::Replacements(); | 
|  | return Cleaner(Environment(Code, FileName, Ranges), Style).process().first; | 
|  | } | 
|  |  | 
|  | tooling::Replacements reformat(const FormatStyle &Style, StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName, bool *IncompleteFormat) { | 
|  | FormattingAttemptStatus Status; | 
|  | auto Result = reformat(Style, Code, Ranges, FileName, &Status); | 
|  | if (!Status.FormatComplete) | 
|  | *IncompleteFormat = true; | 
|  | return Result; | 
|  | } | 
|  |  | 
|  | tooling::Replacements fixNamespaceEndComments(const FormatStyle &Style, | 
|  | StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName) { | 
|  | return NamespaceEndCommentsFixer(Environment(Code, FileName, Ranges), Style) | 
|  | .process() | 
|  | .first; | 
|  | } | 
|  |  | 
|  | tooling::Replacements sortUsingDeclarations(const FormatStyle &Style, | 
|  | StringRef Code, | 
|  | ArrayRef<tooling::Range> Ranges, | 
|  | StringRef FileName) { | 
|  | return UsingDeclarationsSorter(Environment(Code, FileName, Ranges), Style) | 
|  | .process() | 
|  | .first; | 
|  | } | 
|  |  | 
|  | LangOptions getFormattingLangOpts(const FormatStyle &Style) { | 
|  | LangOptions LangOpts; | 
|  |  | 
|  | FormatStyle::LanguageStandard LexingStd = Style.Standard; | 
|  | if (LexingStd == FormatStyle::LS_Auto) | 
|  | LexingStd = FormatStyle::LS_Latest; | 
|  | if (LexingStd == FormatStyle::LS_Latest) | 
|  | LexingStd = FormatStyle::LS_Cpp20; | 
|  | LangOpts.CPlusPlus = 1; | 
|  | LangOpts.CPlusPlus11 = LexingStd >= FormatStyle::LS_Cpp11; | 
|  | LangOpts.CPlusPlus14 = LexingStd >= FormatStyle::LS_Cpp14; | 
|  | LangOpts.CPlusPlus17 = LexingStd >= FormatStyle::LS_Cpp17; | 
|  | LangOpts.CPlusPlus2a = LexingStd >= FormatStyle::LS_Cpp20; | 
|  |  | 
|  | LangOpts.LineComment = 1; | 
|  | bool AlternativeOperators = Style.isCpp(); | 
|  | LangOpts.CXXOperatorNames = AlternativeOperators ? 1 : 0; | 
|  | LangOpts.Bool = 1; | 
|  | LangOpts.ObjC = 1; | 
|  | LangOpts.MicrosoftExt = 1;    // To get kw___try, kw___finally. | 
|  | LangOpts.DeclSpecKeyword = 1; // To get __declspec. | 
|  | return LangOpts; | 
|  | } | 
|  |  | 
|  | const char *StyleOptionHelpDescription = | 
|  | "Coding style, currently supports:\n" | 
|  | "  LLVM, Google, Chromium, Mozilla, WebKit.\n" | 
|  | "Use -style=file to load style configuration from\n" | 
|  | ".clang-format file located in one of the parent\n" | 
|  | "directories of the source file (or current\n" | 
|  | "directory for stdin).\n" | 
|  | "Use -style=\"{key: value, ...}\" to set specific\n" | 
|  | "parameters, e.g.:\n" | 
|  | "  -style=\"{BasedOnStyle: llvm, IndentWidth: 8}\""; | 
|  |  | 
|  | static FormatStyle::LanguageKind getLanguageByFileName(StringRef FileName) { | 
|  | if (FileName.endswith(".java")) | 
|  | return FormatStyle::LK_Java; | 
|  | if (FileName.endswith_lower(".js") || FileName.endswith_lower(".mjs") || | 
|  | FileName.endswith_lower(".ts")) | 
|  | return FormatStyle::LK_JavaScript; // (module) JavaScript or TypeScript. | 
|  | if (FileName.endswith(".m") || FileName.endswith(".mm")) | 
|  | return FormatStyle::LK_ObjC; | 
|  | if (FileName.endswith_lower(".proto") || | 
|  | FileName.endswith_lower(".protodevel")) | 
|  | return FormatStyle::LK_Proto; | 
|  | if (FileName.endswith_lower(".textpb") || | 
|  | FileName.endswith_lower(".pb.txt") || | 
|  | FileName.endswith_lower(".textproto") || | 
|  | FileName.endswith_lower(".asciipb")) | 
|  | return FormatStyle::LK_TextProto; | 
|  | if (FileName.endswith_lower(".td")) | 
|  | return FormatStyle::LK_TableGen; | 
|  | if (FileName.endswith_lower(".cs")) | 
|  | return FormatStyle::LK_CSharp; | 
|  | return FormatStyle::LK_Cpp; | 
|  | } | 
|  |  | 
|  | FormatStyle::LanguageKind guessLanguage(StringRef FileName, StringRef Code) { | 
|  | const auto GuessedLanguage = getLanguageByFileName(FileName); | 
|  | if (GuessedLanguage == FormatStyle::LK_Cpp) { | 
|  | auto Extension = llvm::sys::path::extension(FileName); | 
|  | // If there's no file extension (or it's .h), we need to check the contents | 
|  | // of the code to see if it contains Objective-C. | 
|  | if (Extension.empty() || Extension == ".h") { | 
|  | auto NonEmptyFileName = FileName.empty() ? "guess.h" : FileName; | 
|  | Environment Env(Code, NonEmptyFileName, /*Ranges=*/{}); | 
|  | ObjCHeaderStyleGuesser Guesser(Env, getLLVMStyle()); | 
|  | Guesser.process(); | 
|  | if (Guesser.isObjC()) | 
|  | return FormatStyle::LK_ObjC; | 
|  | } | 
|  | } | 
|  | return GuessedLanguage; | 
|  | } | 
|  |  | 
|  | const char *DefaultFormatStyle = "file"; | 
|  |  | 
|  | const char *DefaultFallbackStyle = "LLVM"; | 
|  |  | 
|  | llvm::Expected<FormatStyle> getStyle(StringRef StyleName, StringRef FileName, | 
|  | StringRef FallbackStyleName, | 
|  | StringRef Code, | 
|  | llvm::vfs::FileSystem *FS) { | 
|  | if (!FS) { | 
|  | FS = llvm::vfs::getRealFileSystem().get(); | 
|  | } | 
|  | FormatStyle Style = getLLVMStyle(guessLanguage(FileName, Code)); | 
|  |  | 
|  | FormatStyle FallbackStyle = getNoStyle(); | 
|  | if (!getPredefinedStyle(FallbackStyleName, Style.Language, &FallbackStyle)) | 
|  | return make_string_error("Invalid fallback style \"" + FallbackStyleName); | 
|  |  | 
|  | if (StyleName.startswith("{")) { | 
|  | // Parse YAML/JSON style from the command line. | 
|  | if (std::error_code ec = parseConfiguration(StyleName, &Style)) | 
|  | return make_string_error("Error parsing -style: " + ec.message()); | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | if (!StyleName.equals_lower("file")) { | 
|  | if (!getPredefinedStyle(StyleName, Style.Language, &Style)) | 
|  | return make_string_error("Invalid value for -style"); | 
|  | return Style; | 
|  | } | 
|  |  | 
|  | // Look for .clang-format/_clang-format file in the file's parent directories. | 
|  | SmallString<128> UnsuitableConfigFiles; | 
|  | SmallString<128> Path(FileName); | 
|  | if (std::error_code EC = FS->makeAbsolute(Path)) | 
|  | return make_string_error(EC.message()); | 
|  |  | 
|  | llvm::SmallVector<std::string, 2> FilesToLookFor; | 
|  | FilesToLookFor.push_back(".clang-format"); | 
|  | FilesToLookFor.push_back("_clang-format"); | 
|  |  | 
|  | for (StringRef Directory = Path; !Directory.empty(); | 
|  | Directory = llvm::sys::path::parent_path(Directory)) { | 
|  |  | 
|  | auto Status = FS->status(Directory); | 
|  | if (!Status || | 
|  | Status->getType() != llvm::sys::fs::file_type::directory_file) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | for (const auto &F : FilesToLookFor) { | 
|  | SmallString<128> ConfigFile(Directory); | 
|  |  | 
|  | llvm::sys::path::append(ConfigFile, F); | 
|  | LLVM_DEBUG(llvm::dbgs() << "Trying " << ConfigFile << "...\n"); | 
|  |  | 
|  | Status = FS->status(ConfigFile.str()); | 
|  |  | 
|  | if (Status && | 
|  | (Status->getType() == llvm::sys::fs::file_type::regular_file)) { | 
|  | llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> Text = | 
|  | FS->getBufferForFile(ConfigFile.str()); | 
|  | if (std::error_code EC = Text.getError()) | 
|  | return make_string_error(EC.message()); | 
|  | if (std::error_code ec = | 
|  | parseConfiguration(Text.get()->getBuffer(), &Style)) { | 
|  | if (ec == ParseError::Unsuitable) { | 
|  | if (!UnsuitableConfigFiles.empty()) | 
|  | UnsuitableConfigFiles.append(", "); | 
|  | UnsuitableConfigFiles.append(ConfigFile); | 
|  | continue; | 
|  | } | 
|  | return make_string_error("Error reading " + ConfigFile + ": " + | 
|  | ec.message()); | 
|  | } | 
|  | LLVM_DEBUG(llvm::dbgs() | 
|  | << "Using configuration file " << ConfigFile << "\n"); | 
|  | return Style; | 
|  | } | 
|  | } | 
|  | } | 
|  | if (!UnsuitableConfigFiles.empty()) | 
|  | return make_string_error("Configuration file(s) do(es) not support " + | 
|  | getLanguageName(Style.Language) + ": " + | 
|  | UnsuitableConfigFiles); | 
|  | return FallbackStyle; | 
|  | } | 
|  |  | 
|  | } // namespace format | 
|  | } // namespace clang |