summaryrefslogtreecommitdiffstats
path: root/tools/gn/command_ls.cc
blob: 8aefa76a08c4d9d081d690d1b549e931af03acfd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// Copyright 2014 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 <algorithm>
#include <set>

#include "base/command_line.h"
#include "tools/gn/commands.h"
#include "tools/gn/label_pattern.h"
#include "tools/gn/setup.h"
#include "tools/gn/standard_out.h"
#include "tools/gn/target.h"

namespace commands {

const char kLs[] = "ls";
const char kLs_HelpShort[] =
    "ls: List matching targets.";
const char kLs_Help[] =
    "gn ls <build dir> [<label_pattern>] [--out] [--all-toolchains]\n"
    "\n"
    "  Lists all targets matching the given pattern for the given builn\n"
    "  directory. By default, only targets in the default toolchain will\n"
    "  be matched unless a toolchain is explicitly supplied.\n"
    "\n"
    "  If the label pattern is unspecified, list all targets. The label\n"
    "  pattern is not a general regular expression (see\n"
    "  \"gn help label_pattern\"). If you need more complex expressions,\n"
    "  pipe the result through grep.\n"
    "\n"
    "  --out\n"
    "      Lists the results as the files generated by the matching targets.\n"
    "      These files will be relative to the build directory such that\n"
    "      they can be specified on Ninja's command line as a file to build.\n"
    "\n"
    "  --all-toolchains\n"
    "      Matches all toolchains. If the label pattern does not specify an\n"
    "      explicit toolchain, labels from all toolchains will be matched.\n"
    "\n"
    "Examples\n"
    "\n"
    "  gn ls out/Debug\n"
    "      Lists all targets in the default toolchain.\n"
    "\n"
    "  gn ls out/Debug \"//base/*\"\n"
    "      Lists all targets in the directory base and all subdirectories.\n"
    "\n"
    "  gn ls out/Debug \"//base:*\"\n"
    "      Lists all targets defined in //base/BUILD.gn.\n"
    "\n"
    "  gn ls out/Debug //base --out\n"
    "      Lists the build output file for //base:base\n"
    "\n"
    "  gn ls out/Debug \"//base/*\" --out | xargs ninja -C out/Debug\n"
    "      Builds all targets in //base and all subdirectories.\n"
    "\n"
    "  gn ls out/Debug //base --all-toolchains\n"
    "      Lists all variants of the target //base:base (it may be referenced\n"
    "      in multiple toolchains).\n";

int RunLs(const std::vector<std::string>& args) {
  if (args.size() != 1 && args.size() != 2) {
    Err(Location(), "You're holding it wrong.",
        "Usage: \"gn ls <build dir> [<label_pattern>]\"").PrintToStdout();
    return 1;
  }

  Setup* setup = new Setup;
  if (!setup->DoSetup(args[0], false) || !setup->Run())
    return 1;

  const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess();
  bool all_toolchains = cmdline->HasSwitch("all-toolchains");

  // Find matching targets.
  std::vector<const Target*> matches;
  if (args.size() == 2) {
    // Given a pattern, match it.
    if (!ResolveTargetsFromCommandLinePattern(setup, args[1], all_toolchains,
                                              &matches))
      return 1;
  } else if (all_toolchains) {
    // List all resolved targets.
    matches = setup->builder()->GetAllResolvedTargets();
  } else {
    // List all resolved targets in the default toolchain.
    for (const auto& target : setup->builder()->GetAllResolvedTargets()) {
      if (target->settings()->is_default())
        matches.push_back(target);
    }
  }

  if (cmdline->HasSwitch("out")) {
    // List results as build files.
    for (const auto& match : matches) {
      OutputString(match->dependency_output_file().value());
      OutputString("\n");
    }
  } else {
    // List results as sorted labels.
    std::vector<Label> sorted_matches;
    for (const auto& match : matches)
      sorted_matches.push_back(match->label());
    std::sort(sorted_matches.begin(), sorted_matches.end());

    Label default_tc_label = setup->loader()->default_toolchain_label();
    for (const auto& match : sorted_matches) {
      // Print toolchain only for ones not in the default toolchain.
      OutputString(match.GetUserVisibleName(
          match.GetToolchainLabel() != default_tc_label));
      OutputString("\n");
    }
  }

  return 0;
}

}  // namespace commands