Skip to content

Commit 98f0783

Browse files
committed
[llvm-strings] Switch command line parsing from llvm::cl to OptTable
Some behavior changes: * `-t=d` is removed. Use `-t d` instead. * one-dash long options like `-all` are supported. Use `--all` instead. * `--all=0` or `--all=false` cannot be used. (Note: `--all` is silently ignored anyway) * `--help-list` is removed. This is a `cl::` specific option. Nobody is likely leveraging any of the above. Advantages: * `-t` diagnostic gets improved. * in the absence of `HideUnrelatedOptions`, `--help` will not list unrelated options if linking against libLLVM-13git.so or linker GC is not used. * Decrease the probability of cl::opt collision if we do decide to support multiplexing Note: because the tool is so simple, used more for forensics instead of a building tool, and its long options are unlikely used in one-dash form, I just drop the one-dash form in this patch. Reviewed By: jhenderson Differential Revision: https://reviews.llvm.org/D104889
1 parent 715137d commit 98f0783

File tree

8 files changed

+136
-45
lines changed

8 files changed

+136
-45
lines changed

llvm/docs/CommandGuide/llvm-strings.rst

-4
Original file line numberDiff line numberDiff line change
@@ -52,10 +52,6 @@ OPTIONS
5252

5353
Display a summary of command line options.
5454

55-
.. option:: --help-list
56-
57-
Display an uncategorized summary of command line options.
58-
5955
.. option:: --print-file-name, -f
6056

6157
Display the name of the containing file before each string.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
## Show that short options can be grouped.
2+
3+
RUN: echo abcd | llvm-strings -af | FileCheck %s
4+
CHECK: {standard input}: abcd
+6-10
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,11 @@
11
## Show that help text is printed correctly when requested.
22

3-
RUN: llvm-strings -h | FileCheck %s --check-prefixes=CHECK,CATEG
4-
RUN: llvm-strings --help | FileCheck %s --check-prefixes=CHECK,CATEG
5-
RUN: llvm-strings --help-list \
6-
RUN: | FileCheck %s --check-prefixes=CHECK,LIST
3+
RUN: llvm-strings -h | FileCheck %s
4+
RUN: llvm-strings --help | FileCheck %s
75

86
CHECK: OVERVIEW: llvm string dumper
9-
CHECK: USAGE: llvm-strings{{(.exe)?}} [options] <input object files>{{$}}
7+
CHECK: USAGE: llvm-strings [options] <input object files>{{$}}
108
CHECK: OPTIONS:
11-
CATEG: General options:
12-
LIST-NOT: General options:
13-
CATEG: Generic Options:
14-
LIST-NOT: Generic Options:
15-
CHECK: @FILE
9+
CHECK: --all
10+
CHECK: -a
11+
CHECK: Pass @FILE as argument to read options from FILE.

llvm/test/tools/llvm-strings/length.test

+3-3
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,9 @@ RUN: llvm-strings --bytes 2 %t | FileCheck --check-prefix CHECK-2 %s --implicit-
2121

2222
## Show different syntaxes work.
2323
RUN: llvm-strings --bytes=2 %t | FileCheck --check-prefix CHECK-2 %s --implicit-check-not={{.}}
24-
RUN: llvm-strings -n=2 %t | FileCheck --check-prefix CHECK-2 %s --implicit-check-not={{.}}
24+
RUN: llvm-strings -n 2 %t | FileCheck --check-prefix CHECK-2 %s --implicit-check-not={{.}}
2525

26-
CHECK-0: invalid minimum string length 0
26+
CHECK-0: llvm-strings: error: expected a positive integer, but got '0'
2727

2828
CHECK-1: a
2929
CHECK-1-NEXT: ab
@@ -43,4 +43,4 @@ CHECK-5: abcde
4343

4444
## Show that a non-numeric argument is rejected.
4545
RUN: not llvm-strings -n foo %t 2>&1 | FileCheck %s --check-prefix=ERR
46-
ERR: llvm-strings{{.*}}: for the --bytes option: 'foo' value invalid for integer argument!
46+
ERR: llvm-strings: error: expected a positive integer, but got 'foo'

llvm/test/tools/llvm-strings/radix.test

+2-2
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ RUN: llvm-strings --radix x %t/a.txt | FileCheck %s -check-prefix CHECK-HEX --st
2626

2727
## Show different syntaxes work.
2828
RUN: llvm-strings --radix=d %t/a.txt | FileCheck %s -check-prefix CHECK-DEC --strict-whitespace
29-
RUN: llvm-strings -t=d %t/a.txt | FileCheck %s -check-prefix CHECK-DEC --strict-whitespace
29+
RUN: llvm-strings -t d %t/a.txt | FileCheck %s -check-prefix CHECK-DEC --strict-whitespace
3030

3131
CHECK-NONE: {{^}}three
3232
CHECK-NONE: {{^}}four
@@ -58,4 +58,4 @@ CHECK-HEX: {{^}} 28 nine
5858

5959
## Show that an invalid value is rejected.
6060
RUN: not llvm-strings --radix z %t/a.txt 2>&1 | FileCheck %s --check-prefix=INVALID
61-
INVALID: llvm-strings{{.*}}: for the --radix option: Cannot find option named 'z'!
61+
INVALID: llvm-strings: error: --radix value should be one of: '' (no offset), 'o' (octal), 'd' (decimal), 'x' (hexadecimal)

llvm/tools/llvm-strings/CMakeLists.txt

+7
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,18 @@
11
set(LLVM_LINK_COMPONENTS
22
Core
33
Object
4+
Option
45
Support
56
)
67

8+
set(LLVM_TARGET_DEFINITIONS Opts.td)
9+
tablegen(LLVM Opts.inc -gen-opt-parser-defs)
10+
add_public_tablegen_target(StringsOptsTableGen)
11+
712
add_llvm_tool(llvm-strings
813
llvm-strings.cpp
14+
DEPENDS
15+
StringsOptsTableGen
916
)
1017

1118
if(LLVM_INSTALL_BINUTILS_SYMLINKS)

llvm/tools/llvm-strings/Opts.td

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
include "llvm/Option/OptParser.td"
2+
3+
class F<string letter, string help> : Flag<["-"], letter>, HelpText<help>;
4+
class FF<string name, string help> : Flag<["--"], name>, HelpText<help>;
5+
6+
multiclass Eq<string name, string help> {
7+
def NAME #_EQ : Joined<["--"], name #"=">,
8+
HelpText<help>;
9+
def : Separate<["--"], name>, Alias<!cast<Joined>(NAME #_EQ)>;
10+
}
11+
12+
def all : FF<"all", "Silently ignored. Present for GNU strings compatibility">;
13+
defm bytes : Eq<"bytes", "Print sequences of the specified length">;
14+
def help : FF<"help", "Display this help">;
15+
def print_file_name : Flag<["--"], "print-file-name">, HelpText<"Print the name of the file before each string">;
16+
defm radix : Eq<"radix", "Print the offset within the file with the specified radix: o (octal), d (decimal), x (hexadecimal)">, MetaVarName<"<radix>">;
17+
def version : FF<"version", "Display the version">;
18+
19+
def : F<"a", "Alias for --all">, Alias<all>;
20+
def : F<"f", "Alias for --print-file-name">, Alias<print_file_name>;
21+
def : F<"h", "Alias for --help">, Alias<help>;
22+
def : JoinedOrSeparate<["-"], "n">, Alias<bytes_EQ>, HelpText<"Alias for --bytes">;
23+
def : JoinedOrSeparate<["-"], "t">, Alias<radix_EQ>, HelpText<"Alias for --radix">, MetaVarName<"<radix>">;

llvm/tools/llvm-strings/llvm-strings.cpp

+91-26
Original file line numberDiff line numberDiff line change
@@ -11,51 +11,81 @@
1111
//
1212
//===----------------------------------------------------------------------===//
1313

14+
#include "Opts.inc"
1415
#include "llvm/Object/Binary.h"
16+
#include "llvm/Option/Arg.h"
17+
#include "llvm/Option/ArgList.h"
18+
#include "llvm/Option/Option.h"
1519
#include "llvm/Support/CommandLine.h"
1620
#include "llvm/Support/Error.h"
1721
#include "llvm/Support/Format.h"
1822
#include "llvm/Support/InitLLVM.h"
1923
#include "llvm/Support/MemoryBuffer.h"
2024
#include "llvm/Support/Program.h"
25+
#include "llvm/Support/WithColor.h"
2126
#include <cctype>
2227
#include <string>
2328

2429
using namespace llvm;
2530
using namespace llvm::object;
2631

32+
namespace {
33+
enum ID {
34+
OPT_INVALID = 0, // This is not an option ID.
35+
#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
36+
HELPTEXT, METAVAR, VALUES) \
37+
OPT_##ID,
38+
#include "Opts.inc"
39+
#undef OPTION
40+
};
41+
42+
#define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
43+
#include "Opts.inc"
44+
#undef PREFIX
45+
46+
static const opt::OptTable::Info InfoTable[] = {
47+
#define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
48+
HELPTEXT, METAVAR, VALUES) \
49+
{ \
50+
PREFIX, NAME, HELPTEXT, \
51+
METAVAR, OPT_##ID, opt::Option::KIND##Class, \
52+
PARAM, FLAGS, OPT_##GROUP, \
53+
OPT_##ALIAS, ALIASARGS, VALUES},
54+
#include "Opts.inc"
55+
#undef OPTION
56+
};
57+
58+
class StringsOptTable : public opt::OptTable {
59+
public:
60+
StringsOptTable() : OptTable(InfoTable) { setGroupedShortOptions(true); }
61+
};
62+
} // namespace
63+
64+
const char ToolName[] = "llvm-strings";
65+
2766
static cl::list<std::string> InputFileNames(cl::Positional,
2867
cl::desc("<input object files>"),
2968
cl::ZeroOrMore);
3069

31-
static cl::opt<bool>
32-
PrintFileName("print-file-name",
33-
cl::desc("Print the name of the file before each string"));
34-
static cl::alias PrintFileNameShort("f", cl::desc(""),
35-
cl::aliasopt(PrintFileName));
36-
37-
static cl::opt<int>
38-
MinLength("bytes", cl::desc("Print sequences of the specified length"),
39-
cl::init(4));
40-
static cl::alias MinLengthShort("n", cl::desc(""), cl::aliasopt(MinLength));
41-
42-
static cl::opt<bool>
43-
AllSections("all",
44-
cl::desc("Check all sections, not just the data section"));
45-
static cl::alias AllSectionsShort("a", cl::desc(""),
46-
cl::aliasopt(AllSections));
70+
static int MinLength = 4;
71+
static bool PrintFileName;
4772

4873
enum radix { none, octal, hexadecimal, decimal };
49-
static cl::opt<radix>
50-
Radix("radix", cl::desc("print the offset within the file"),
51-
cl::values(clEnumValN(octal, "o", "octal"),
52-
clEnumValN(hexadecimal, "x", "hexadecimal"),
53-
clEnumValN(decimal, "d", "decimal")),
54-
cl::init(none));
55-
static cl::alias RadixShort("t", cl::desc(""), cl::aliasopt(Radix));
74+
static radix Radix;
5675

57-
static cl::extrahelp
58-
HelpResponse("\nPass @FILE as argument to read options from FILE.\n");
76+
LLVM_ATTRIBUTE_NORETURN static void reportCmdLineError(const Twine &Message) {
77+
WithColor::error(errs(), ToolName) << Message << "\n";
78+
exit(1);
79+
}
80+
81+
template <typename T>
82+
static void parseIntArg(const opt::InputArgList &Args, int ID, T &Value) {
83+
if (const opt::Arg *A = Args.getLastArg(ID)) {
84+
StringRef V(A->getValue());
85+
if (!llvm::to_integer(V, Value, 0) || Value <= 0)
86+
reportCmdLineError("expected a positive integer, but got '" + V + "'");
87+
}
88+
}
5989

6090
static void strings(raw_ostream &OS, StringRef FileName, StringRef Contents) {
6191
auto print = [&OS, FileName](unsigned Offset, StringRef L) {
@@ -96,13 +126,48 @@ static void strings(raw_ostream &OS, StringRef FileName, StringRef Contents) {
96126

97127
int main(int argc, char **argv) {
98128
InitLLVM X(argc, argv);
129+
BumpPtrAllocator A;
130+
StringSaver Saver(A);
131+
StringsOptTable Tbl;
132+
opt::InputArgList Args =
133+
Tbl.parseArgs(argc, argv, OPT_UNKNOWN, Saver,
134+
[&](StringRef Msg) { reportCmdLineError(Msg); });
135+
if (Args.hasArg(OPT_help)) {
136+
Tbl.printHelp(
137+
outs(),
138+
(Twine(ToolName) + " [options] <input object files>").str().c_str(),
139+
"llvm string dumper");
140+
// TODO Replace this with OptTable API once it adds extrahelp support.
141+
outs() << "\nPass @FILE as argument to read options from FILE.\n";
142+
return 0;
143+
}
144+
if (Args.hasArg(OPT_version)) {
145+
outs() << ToolName << '\n';
146+
cl::PrintVersionMessage();
147+
return 0;
148+
}
149+
150+
parseIntArg(Args, OPT_bytes_EQ, MinLength);
151+
PrintFileName = Args.hasArg(OPT_print_file_name);
152+
StringRef R = Args.getLastArgValue(OPT_radix_EQ);
153+
if (R.empty())
154+
Radix = none;
155+
else if (R == "o")
156+
Radix = octal;
157+
else if (R == "d")
158+
Radix = decimal;
159+
else if (R == "x")
160+
Radix = hexadecimal;
161+
else
162+
reportCmdLineError("--radix value should be one of: '' (no offset), 'o' "
163+
"(octal), 'd' (decimal), 'x' (hexadecimal)");
99164

100-
cl::ParseCommandLineOptions(argc, argv, "llvm string dumper\n");
101165
if (MinLength == 0) {
102166
errs() << "invalid minimum string length 0\n";
103167
return EXIT_FAILURE;
104168
}
105169

170+
std::vector<std::string> InputFileNames = Args.getAllArgValues(OPT_INPUT);
106171
if (InputFileNames.empty())
107172
InputFileNames.push_back("-");
108173

0 commit comments

Comments
 (0)