• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "tools/gn/standard_out.h"
6 
7 #include <vector>
8 
9 #include "base/logging.h"
10 #include "base/strings/string_split.h"
11 #include "build/build_config.h"
12 
13 #if defined(OS_WIN)
14 #include <windows.h>
15 #else
16 #include <stdio.h>
17 #include <unistd.h>
18 #endif
19 
20 namespace {
21 
22 bool initialized = false;
23 
24 #if defined(OS_WIN)
25 HANDLE hstdout;
26 WORD default_attributes;
27 #endif
28 bool is_console = false;
29 
EnsureInitialized()30 void EnsureInitialized() {
31   if (initialized)
32     return;
33   initialized = true;
34 
35 #if defined(OS_WIN)
36   hstdout = ::GetStdHandle(STD_OUTPUT_HANDLE);
37   CONSOLE_SCREEN_BUFFER_INFO info;
38   is_console = !!::GetConsoleScreenBufferInfo(hstdout, &info);
39   default_attributes = info.wAttributes;
40 #else
41   is_console = isatty(fileno(stdout));
42 #endif
43 }
44 
WriteToStdOut(const std::string & output)45 void WriteToStdOut(const std::string& output) {
46   size_t written_bytes = fwrite(output.data(), 1, output.size(), stdout);
47   DCHECK_EQ(output.size(), written_bytes);
48 }
49 
50 }  // namespace
51 
52 #if defined(OS_WIN)
53 
OutputString(const std::string & output,TextDecoration dec)54 void OutputString(const std::string& output, TextDecoration dec) {
55   EnsureInitialized();
56   if (is_console) {
57     switch (dec) {
58       case DECORATION_NONE:
59         break;
60       case DECORATION_DIM:
61         ::SetConsoleTextAttribute(hstdout, FOREGROUND_INTENSITY);
62         break;
63       case DECORATION_RED:
64         ::SetConsoleTextAttribute(hstdout,
65                                   FOREGROUND_RED | FOREGROUND_INTENSITY);
66         break;
67       case DECORATION_GREEN:
68         // Keep green non-bold.
69         ::SetConsoleTextAttribute(hstdout, FOREGROUND_GREEN);
70         break;
71       case DECORATION_BLUE:
72         ::SetConsoleTextAttribute(hstdout,
73                                   FOREGROUND_BLUE | FOREGROUND_INTENSITY);
74         break;
75       case DECORATION_YELLOW:
76         ::SetConsoleTextAttribute(hstdout,
77                                   FOREGROUND_RED | FOREGROUND_GREEN);
78         break;
79     }
80   }
81 
82   DWORD written = 0;
83   ::WriteFile(hstdout, output.c_str(), static_cast<DWORD>(output.size()),
84               &written, NULL);
85 
86   if (is_console)
87     ::SetConsoleTextAttribute(hstdout, default_attributes);
88 }
89 
90 #else
91 
OutputString(const std::string & output,TextDecoration dec)92 void OutputString(const std::string& output, TextDecoration dec) {
93   EnsureInitialized();
94   if (is_console) {
95     switch (dec) {
96       case DECORATION_NONE:
97         break;
98       case DECORATION_DIM:
99         WriteToStdOut("\e[2m");
100         break;
101       case DECORATION_RED:
102         WriteToStdOut("\e[31m\e[1m");
103         break;
104       case DECORATION_GREEN:
105         WriteToStdOut("\e[32m");
106         break;
107       case DECORATION_BLUE:
108         WriteToStdOut("\e[34m\e[1m");
109         break;
110       case DECORATION_YELLOW:
111         WriteToStdOut("\e[33m\e[1m");
112         break;
113     }
114   }
115 
116   WriteToStdOut(output.data());
117 
118   if (dec != DECORATION_NONE)
119     WriteToStdOut("\e[0m");
120 }
121 
122 #endif
123 
PrintShortHelp(const std::string & line)124 void PrintShortHelp(const std::string& line) {
125   size_t colon_offset = line.find(':');
126   size_t first_normal = 0;
127   if (colon_offset != std::string::npos) {
128     OutputString("  " + line.substr(0, colon_offset), DECORATION_YELLOW);
129     first_normal = colon_offset;
130   }
131 
132   // See if the colon is followed by a " [" and if so, dim the contents of [ ].
133   if (first_normal > 0 &&
134       line.size() > first_normal + 2 &&
135       line[first_normal + 1] == ' ' && line[first_normal + 2] == '[') {
136     size_t begin_bracket = first_normal + 2;
137     OutputString(": ");
138     first_normal = line.find(']', begin_bracket);
139     if (first_normal == std::string::npos)
140       first_normal = line.size();
141     else
142       first_normal++;
143     OutputString(line.substr(begin_bracket, first_normal - begin_bracket),
144                  DECORATION_DIM);
145   }
146 
147   OutputString(line.substr(first_normal) + "\n");
148 }
149 
PrintLongHelp(const std::string & text)150 void PrintLongHelp(const std::string& text) {
151   std::vector<std::string> lines;
152   base::SplitStringDontTrim(text, '\n', &lines);
153 
154   for (size_t i = 0; i < lines.size(); i++) {
155     const std::string& line = lines[i];
156 
157     // Check for a heading line.
158     if (!line.empty() && line[0] != ' ') {
159       // Highlight up to the colon (if any).
160       size_t chars_to_highlight = line.find(':');
161       if (chars_to_highlight == std::string::npos)
162         chars_to_highlight = line.size();
163       OutputString(line.substr(0, chars_to_highlight), DECORATION_YELLOW);
164       OutputString(line.substr(chars_to_highlight) + "\n");
165       continue;
166     }
167 
168     // Check for a comment.
169     TextDecoration dec = DECORATION_NONE;
170     for (size_t char_i = 0; char_i < line.size(); char_i++) {
171       if (line[char_i] == '#') {
172         // Got a comment, draw dimmed.
173         dec = DECORATION_DIM;
174         break;
175       } else if (line[char_i] != ' ') {
176         break;
177       }
178     }
179 
180     OutputString(line + "\n", dec);
181   }
182 }
183 
184