// Copyright (c) 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include #include "base/command_line.h" #include "tools/gn/args.h" #include "tools/gn/commands.h" #include "tools/gn/err.h" #include "tools/gn/functions.h" #include "tools/gn/input_conversion.h" #include "tools/gn/label.h" #include "tools/gn/label_pattern.h" #include "tools/gn/ninja_build_writer.h" #include "tools/gn/parser.h" #include "tools/gn/runtime_deps.h" #include "tools/gn/setup.h" #include "tools/gn/standard_out.h" #include "tools/gn/string_utils.h" #include "tools/gn/substitution_writer.h" #include "tools/gn/switches.h" #include "tools/gn/target.h" #include "tools/gn/variables.h" namespace commands { namespace { void PrintToplevelHelp() { PrintSectionHelp("Commands", "", "commands"); for (const auto& cmd : commands::GetCommands()) PrintShortHelp(cmd.second.help_short); // Target declarations. PrintSectionHelp("Target declarations", "", "targets"); for (const auto& func : functions::GetFunctions()) { if (func.second.is_target) PrintShortHelp(func.second.help_short); } // Functions. PrintSectionHelp("Buildfile functions", "", "functions"); for (const auto& func : functions::GetFunctions()) { if (!func.second.is_target) PrintShortHelp(func.second.help_short); } // Built-in variables. PrintSectionHelp("Built-in predefined variables", "", "predefined_variables"); for (const auto& builtin : variables::GetBuiltinVariables()) PrintShortHelp(builtin.second.help_short); // Target variables. PrintSectionHelp("Variables you set in targets", "", "target_variables"); for (const auto& target : variables::GetTargetVariables()) PrintShortHelp(target.second.help_short); PrintSectionHelp("Other help topics", "", "other"); PrintShortHelp("all: Print all the help at once"); PrintShortHelp("buildargs: How build arguments work."); PrintShortHelp("dotfile: Info about the toplevel .gn file."); PrintShortHelp("execution: Build graph and execution overview."); PrintShortHelp("grammar: Language and grammar for GN build files."); PrintShortHelp( "input_conversion: Processing input from exec_script and read_file."); PrintShortHelp("label_pattern: Matching more than one label."); PrintShortHelp("labels: About labels."); PrintShortHelp("ninja_rules: How Ninja build rules are named."); PrintShortHelp("nogncheck: Annotating includes for checking."); PrintShortHelp("runtime_deps: How runtime dependency computation works."); PrintShortHelp("source_expansion: Map sources to outputs for scripts."); PrintShortHelp("switches: Show available command-line switches."); } void PrintSwitchHelp() { const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess(); bool is_markdown = cmdline->HasSwitch(switches::kMarkdown); OutputString("Available global switches\n", DECORATION_YELLOW); OutputString( " Do \"gn help --the_switch_you_want_help_on\" for more. Individual\n" " commands may take command-specific switches not listed here. See the\n" " help on your specific command for more.\n\n"); if (is_markdown) OutputString("```\n", DECORATION_NONE); for (const auto& s : switches::GetSwitches()) PrintShortHelp(s.second.short_help); if (is_markdown) OutputString("```\n", DECORATION_NONE); OutputString("\n"); } void PrintAllHelp() { const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess(); bool is_markdown = cmdline->HasSwitch(switches::kMarkdown); if (is_markdown) { OutputString("# GN Reference\n\n"); OutputString("*This page is automatically generated from* " "`gn help --markdown all`.\n\n"); // Generate our own table of contents so that we have more control // over what's in and out. OutputString("## Contents\n\n"); } PrintToplevelHelp(); OutputString("\n"); if (is_markdown) OutputString("## Commands\n\n"); for (const auto& c: commands::GetCommands()) PrintLongHelp(c.second.help); if (is_markdown) OutputString("## Target declarations\n\n"); for (const auto& f : functions::GetFunctions()) { if (f.second.is_target) PrintLongHelp(f.second.help); } if (is_markdown) OutputString("## Buildfile functions\n\n"); for (const auto& f : functions::GetFunctions()) { if (!f.second.is_target) PrintLongHelp(f.second.help); } if (is_markdown) { OutputString( "## " "Built-in predefined variables\n\n"); } for (const auto& v: variables::GetBuiltinVariables()) PrintLongHelp(v.second.help); if (is_markdown) { OutputString( "## " "Variables you set in targets\n\n"); } for (const auto& v: variables::GetTargetVariables()) PrintLongHelp(v.second.help); if (is_markdown) OutputString("## Other help topics\n\n"); PrintLongHelp(kBuildArgs_Help, "buildargs"); PrintLongHelp(kDotfile_Help, "dotfile"); PrintLongHelp(kExecution_Help, "execution"); PrintLongHelp(kGrammar_Help, "grammar"); PrintLongHelp(kInputConversion_Help, "input_conversion"); PrintLongHelp(kLabelPattern_Help, "label_pattern"); PrintLongHelp(kLabels_Help, "labels"); PrintLongHelp(kNinjaRules_Help, "ninja_rules"); PrintLongHelp(kNoGnCheck_Help, "nogncheck"); PrintLongHelp(kRuntimeDeps_Help, "runtime_deps"); PrintLongHelp(kSourceExpansion_Help, "source_expansion"); if (is_markdown) OutputString("## Command Line Switches\n\n"); PrintSwitchHelp(); } // Prints help on the given switch. There should be no leading hyphens. Returns // true if the switch was found and help was printed. False means the switch is // unknown. bool PrintHelpOnSwitch(const std::string& what) { const switches::SwitchInfoMap& all = switches::GetSwitches(); switches::SwitchInfoMap::const_iterator found = all.find(base::StringPiece(what)); if (found == all.end()) return false; PrintLongHelp(found->second.long_help); return true; } } // namespace const char kHelp[] = "help"; const char kHelp_HelpShort[] = "help: Does what you think."; const char kHelp_Help[] = R"(gn help Yo dawg, I heard you like help on your help so I put help on the help in the help. You can also use "all" as the parameter to get all help at once. Switches --markdown Format output in markdown syntax. Example gn help --markdown all Dump all help to stdout in markdown format. )"; int RunHelp(const std::vector& args) { std::string what; if (args.size() == 0) { // If no argument is specified, check for switches to allow things like // "gn help --args" for help on the args switch. const base::CommandLine::SwitchMap& switches = base::CommandLine::ForCurrentProcess()->GetSwitches(); if (switches.empty()) { // Still nothing, show help overview. PrintToplevelHelp(); return 0; } // Switch help needs to be done separately. The CommandLine will strip the // switch separators so --args will come out as "args" which is then // ambiguous with the variable named "args". if (!PrintHelpOnSwitch(switches.begin()->first)) PrintToplevelHelp(); return 0; } else { what = args[0]; } std::vector all_help_topics; // Check commands. const commands::CommandInfoMap& command_map = commands::GetCommands(); auto found_command = command_map.find(what); if (found_command != command_map.end()) { PrintLongHelp(found_command->second.help); return 0; } for (const auto& entry : command_map) all_help_topics.push_back(entry.first); // Check functions. const functions::FunctionInfoMap& function_map = functions::GetFunctions(); auto found_function = function_map.find(what); if (found_function != function_map.end()) PrintLongHelp(found_function->second.help); for (const auto& entry : function_map) all_help_topics.push_back(entry.first); // Builtin variables. const variables::VariableInfoMap& builtin_vars = variables::GetBuiltinVariables(); auto found_builtin_var = builtin_vars.find(what); if (found_builtin_var != builtin_vars.end()) PrintLongHelp(found_builtin_var->second.help); for (const auto& entry : builtin_vars) all_help_topics.push_back(entry.first); // Target variables. const variables::VariableInfoMap& target_vars = variables::GetTargetVariables(); auto found_target_var = target_vars.find(what); if (found_target_var != target_vars.end()) PrintLongHelp(found_target_var->second.help); for (const auto& entry : target_vars) all_help_topics.push_back(entry.first); if (found_function != function_map.end() || found_builtin_var != builtin_vars.end() || found_target_var != target_vars.end()) return 0; // Random other topics. std::map random_topics; random_topics["all"] = PrintAllHelp; random_topics["execution"] = []() { PrintLongHelp(kExecution_Help); }; random_topics["buildargs"] = []() { PrintLongHelp(kBuildArgs_Help); }; random_topics["dotfile"] = []() { PrintLongHelp(kDotfile_Help); }; random_topics["grammar"] = []() { PrintLongHelp(kGrammar_Help); }; random_topics["input_conversion"] = []() { PrintLongHelp(kInputConversion_Help); }; random_topics["label_pattern"] = []() { PrintLongHelp(kLabelPattern_Help); }; random_topics["labels"] = []() { PrintLongHelp(kLabels_Help); }; random_topics["ninja_rules"] = []() { PrintLongHelp(kNinjaRules_Help); }; random_topics["nogncheck"] = []() { PrintLongHelp(kNoGnCheck_Help); }; random_topics["runtime_deps"] = []() { PrintLongHelp(kRuntimeDeps_Help); }; random_topics["source_expansion"] = []() { PrintLongHelp(kSourceExpansion_Help); }; random_topics["switches"] = PrintSwitchHelp; auto found_random_topic = random_topics.find(what); if (found_random_topic != random_topics.end()) { found_random_topic->second(); return 0; } for (const auto& entry : random_topics) all_help_topics.push_back(entry.first); // No help on this. Err(Location(), "No help on \"" + what + "\".").PrintToStdout(); base::StringPiece suggestion = SpellcheckString(what, all_help_topics); if (suggestion.empty()) { OutputString("Run `gn help` for a list of available topics.\n", DECORATION_NONE); } else { OutputString("Did you mean `gn help " + suggestion.as_string() + "`?\n", DECORATION_NONE); } return 1; } } // namespace commands