1
0
mirror of https://github.com/RPCS3/llvm-mirror.git synced 2024-10-19 02:52:53 +02:00

[llvm-objcopy] Allow regular expressions in name comparison

Differential revision: https://reviews.llvm.org/D57517

llvm-svn: 353289
This commit is contained in:
Eugene Leviant 2019-02-06 11:00:07 +00:00
parent ff36ab6ee3
commit f225583f53
14 changed files with 175 additions and 27 deletions

View File

@ -4,6 +4,8 @@
# RUN: --globalize-symbol Weak \ # RUN: --globalize-symbol Weak \
# RUN: --globalize-symbol WeakUndef %t %t2 # RUN: --globalize-symbol WeakUndef %t %t2
# RUN: llvm-readobj --symbols %t2 | FileCheck %s # RUN: llvm-readobj --symbols %t2 | FileCheck %s
# RUN: llvm-objcopy --regex --globalize-symbol '.*' %t %t3
# RUN: cmp %t2 %t3
!ELF !ELF
FileHeader: FileHeader:

View File

@ -28,6 +28,7 @@
# RUN: echo "Unknown" >> %t-globals2.txt # RUN: echo "Unknown" >> %t-globals2.txt
# RUN: echo " " >> %t-globals2.txt # RUN: echo " " >> %t-globals2.txt
# RUN: echo "# File with no symbols" > %t-globals3.txt # RUN: echo "# File with no symbols" > %t-globals3.txt
# RUN: echo "^Global[0-9]+([[:space:]]Global6)*$" > %t-globals-regexp.txt
# RUN: llvm-objcopy \ # RUN: llvm-objcopy \
# RUN: -G Global1 \ # RUN: -G Global1 \
@ -41,6 +42,9 @@
# RUN: %t.o %t.2.o # RUN: %t.o %t.2.o
# RUN: llvm-readobj --elf-output-style=GNU --symbols %t.2.o | FileCheck %s # RUN: llvm-readobj --elf-output-style=GNU --symbols %t.2.o | FileCheck %s
# RUN: llvm-objcopy --regex --keep-global-symbols %t-globals-regexp.txt %t.o %t.3.o
# RUN: llvm-readobj --elf-output-style=GNU --symbols %t.3.o | FileCheck %s --check-prefix=REGEXP
!ELF !ELF
FileHeader: FileHeader:
Class: ELFCLASS64 Class: ELFCLASS64
@ -97,3 +101,12 @@ Symbols:
# CHECK-NEXT: 11: {{.*}} GLOBAL {{.*}} UND Global7 # CHECK-NEXT: 11: {{.*}} GLOBAL {{.*}} UND Global7
# CHECK-NEXT: 12: {{.*}} WEAK {{.*}} Weak1 # CHECK-NEXT: 12: {{.*}} WEAK {{.*}} Weak1
# CHECK-NEXT: 13: {{.*}} GLOBAL {{.*}} Weak2 # CHECK-NEXT: 13: {{.*}} GLOBAL {{.*}} Weak2
# REGEXP: 6: {{.*}} GLOBAL {{.*}} Global1
# REGEXP-NEXT: 7: {{.*}} GLOBAL {{.*}} Global2
# REGEXP-NEXT: 8: {{.*}} GLOBAL {{.*}} Global3
# REGEXP-NEXT: 9: {{.*}} GLOBAL {{.*}} Global4
# REGEXP-NEXT: 10: {{.*}} GLOBAL {{.*}} Global5
# REGEXP-NEXT: 11: {{.*}} GLOBAL {{.*}} Global6
# REGEXP-NEXT: 12: {{.*}} GLOBAL {{.*}} Global5 Global6
# REGEXP-NEXT: 13: {{.*}} GLOBAL {{.*}} UND Global7

View File

@ -1,6 +1,8 @@
# RUN: yaml2obj %s > %t # RUN: yaml2obj %s > %t
# RUN: llvm-objcopy --strip-non-alloc --keep-section=.test --keep-section=.test3 %t %t2 # RUN: llvm-objcopy --strip-non-alloc --keep-section=.test --keep-section=.test3 %t %t2
# RUN: llvm-objcopy --strip-non-alloc --regex --keep-section='^.test[0-9]+$' %t %t3
# RUN: llvm-readobj --file-headers --sections %t2 | FileCheck %s # RUN: llvm-readobj --file-headers --sections %t2 | FileCheck %s
# RUN: llvm-readobj --file-headers --sections %t3 | FileCheck %s --check-prefix=REGEX
!ELF !ELF
FileHeader: FileHeader:
@ -25,3 +27,8 @@ Sections:
# CHECK: Name: .test # CHECK: Name: .test
# CHECK: Name: .test3 # CHECK: Name: .test3
# CHECK: Name: .shstrtab # CHECK: Name: .shstrtab
# REGEX: SectionHeaderCount: 4
# REGEX: Name: .test2
# REGEX: Name: .test3
# REGEX: Name: .shstrtab

View File

@ -3,6 +3,8 @@
# RUN: llvm-objcopy -j .test --keep-section=.test2 %t %t3 # RUN: llvm-objcopy -j .test --keep-section=.test2 %t %t3
# RUN: llvm-readobj --file-headers --sections %t2 | FileCheck %s # RUN: llvm-readobj --file-headers --sections %t2 | FileCheck %s
# RUN: diff %t2 %t3 # RUN: diff %t2 %t3
# RUN: llvm-objcopy --regex --keep-section='^.test$' --only-section='^.test[2-3]+$' %t %t4
# RUN: llvm-readobj --file-headers --sections %t4 | FileCheck %s --check-prefix=REGEX
!ELF !ELF
FileHeader: FileHeader:
@ -25,3 +27,12 @@ Sections:
# CHECK: Name: .symtab # CHECK: Name: .symtab
# CHECK: Name: .strtab # CHECK: Name: .strtab
# CHECK: Name: .shstrtab # CHECK: Name: .shstrtab
# REGEX: SectionHeaderCount: 7
# REGEX: Name: .test
# REGEX: Name: .test2
# REGEX: Name: .test3
# REGEX: Name: .symtab
# REGEX: Name: .strtab
# REGEX: Name: .shstrtab

View File

@ -1,8 +1,10 @@
# RUN: yaml2obj %s > %t # RUN: yaml2obj %s > %t
# RUN: llvm-objcopy --discard-all -K foo --keep-symbol bar %t %t2 # RUN: llvm-objcopy --discard-all -K foo --keep-symbol bar %t %t2
# RUN: llvm-objcopy -K foo -N foo -N bar --keep-symbol bar -N baz %t %t3 # RUN: llvm-objcopy -K foo -N foo -N bar --keep-symbol bar -N baz %t %t3
# RUN: llvm-objcopy --discard-all --regex -K '^ba.*' %t %t4
# RUN: llvm-readobj --symbols %t2 | FileCheck %s # RUN: llvm-readobj --symbols %t2 | FileCheck %s
# RUN: llvm-readobj --symbols %t3 | FileCheck %s # RUN: llvm-readobj --symbols %t3 | FileCheck %s
# RUN: llvm-readobj --symbols %t4 | FileCheck %s --check-prefix=REGEX
!ELF !ELF
FileHeader: FileHeader:
@ -64,3 +66,9 @@ Symbols:
#CHECK-NEXT: Section: .text #CHECK-NEXT: Section: .text
#CHECK-NEXT: } #CHECK-NEXT: }
#CHECK-NEXT:] #CHECK-NEXT:]
#REGEX: Symbols [
#REGEX-NOT: foo
#REGEX: Name: bar
#REGEX: Name: baz
#REGEX: ]

View File

@ -7,6 +7,8 @@
# RUN: -L GlobalCommon \ # RUN: -L GlobalCommon \
# RUN: %t %t2 # RUN: %t %t2
# RUN: llvm-readobj --symbols %t2 | FileCheck %s # RUN: llvm-readobj --symbols %t2 | FileCheck %s
# RUN: llvm-objcopy --regex -L '.*' %t %t3
# RUN: cmp %t2 %t3
!ELF !ELF
FileHeader: FileHeader:

View File

@ -0,0 +1,60 @@
# RUN: yaml2obj %s > %t
# No symbol matches, because pattern is implicitly enveloped in '^$'
# RUN: llvm-objcopy --discard-all --regex -K 'ba' %t %t2
# RUN: llvm-readobj --symbols %t2 | FileCheck %s --check-prefix=REGEX1
# Symbol 'bar' matches
# RUN: llvm-objcopy --discard-all --regex -K 'ba.*' %t %t3
# RUN: llvm-readobj --symbols %t3 | FileCheck %s --check-prefix=REGEX2
# All symbols match
# RUN: llvm-objcopy --discard-all --regex -K '.*ba.*' %t %t4
# RUN: llvm-readobj --symbols %t4 | FileCheck %s --check-prefix=REGEX3
# All symbols match
# RUN: llvm-objcopy --discard-all --regex -K '^([a-z]+)*ba.*$' %t %t5
# RUN: cmp %t4 %t5
!ELF
FileHeader:
Class: ELFCLASS64
Data: ELFDATA2LSB
Type: ET_REL
Machine: EM_X86_64
Sections:
- Name: .text
Type: SHT_PROGBITS
Flags: [ SHF_ALLOC, SHF_EXECINSTR ]
Address: 0x1000
AddressAlign: 0x0000000000000010
Size: 64
Symbols:
Local:
- Name: foobaz
Type: STT_FUNC
Section: .text
Value: 0x1000
Size: 8
- Name: bar
Type: STT_FUNC
Section: .text
Value: 0x1008
Size: 8
- Name: rebar
Type: STT_FUNC
Section: .text
Value: 0x1010
Size: 8
#REGEX1-NOT: foobaz
#REGEX1-NOT: bar
#REGEX1-NOT: rebar
#REGEX2-NOT: foobaz
#REGEX2: bar
#REGEX2-NOT: rebar
#REGEX3: foobaz
#REGEX3: bar
#REGEX3: rebar

View File

@ -11,6 +11,10 @@
# RUN: llvm-objcopy --remove-section=.test2 %t1 %t3 # RUN: llvm-objcopy --remove-section=.test2 %t1 %t3
# RUN: cmp %t2 %t3 # RUN: cmp %t2 %t3
# RUN: llvm-objcopy --regex --remove-section='.test[0-9]+' %t %t4
# RUN: llvm-readobj --file-headers --sections %t4 | FileCheck %s --check-prefix=REGEX
!ELF !ELF
FileHeader: FileHeader:
Class: ELFCLASS64 Class: ELFCLASS64
@ -116,3 +120,7 @@ Sections:
# CHECK: EntrySize: 0 # CHECK: EntrySize: 0
# CHECK: } # CHECK: }
# CHECK: ] # CHECK: ]
# REGEX: SectionHeaderCount: 4
# REGEX: Sections [
# REGEX-NOT: test

View File

@ -2,7 +2,9 @@
# RUN: llvm-objcopy --strip-symbol baz -N bar %t %t2 # RUN: llvm-objcopy --strip-symbol baz -N bar %t %t2
# RUN: llvm-readobj --symbols --sections %t2 | FileCheck %s # RUN: llvm-readobj --symbols --sections %t2 | FileCheck %s
# RUN: llvm-strip --strip-symbol baz -N bar %t -o %t3 # RUN: llvm-strip --strip-symbol baz -N bar %t -o %t3
# RUN: llvm-readobj --symbols --sections %t3 | FileCheck %s # RUN: cmp %t2 %t3
# RUN: llvm-strip --regex --strip-symbol '^b.*' -N bar %t -o %t4
# RUN: cmp %t3 %t4
!ELF !ELF
FileHeader: FileHeader:

View File

@ -1,6 +1,8 @@
# RUN: yaml2obj %s > %t # RUN: yaml2obj %s > %t
# RUN: llvm-objcopy --weaken-symbol Global -W Local -W Weak %t %t2 # RUN: llvm-objcopy --weaken-symbol Global -W Local -W Weak %t %t2
# RUN: llvm-readobj --symbols %t2 | FileCheck %s # RUN: llvm-readobj --symbols %t2 | FileCheck %s
# RUN: llvm-objcopy --regex --weaken-symbol '.*' %t %t3
# RUN: cmp %t2 %t3
!ELF !ELF
FileHeader: FileHeader:

View File

@ -225,9 +225,9 @@ static const MachineInfo &getOutputFormatMachineInfo(StringRef Format) {
return Iter->getValue(); return Iter->getValue();
} }
static void addGlobalSymbolsFromFile(std::vector<StringRef> &Symbols, static void addGlobalSymbolsFromFile(std::vector<NameOrRegex> &Symbols,
BumpPtrAllocator &Alloc, BumpPtrAllocator &Alloc,
StringRef Filename) { StringRef Filename, bool UseRegex) {
StringSaver Saver(Alloc); StringSaver Saver(Alloc);
SmallVector<StringRef, 16> Lines; SmallVector<StringRef, 16> Lines;
auto BufOrErr = MemoryBuffer::getFile(Filename); auto BufOrErr = MemoryBuffer::getFile(Filename);
@ -240,10 +240,21 @@ static void addGlobalSymbolsFromFile(std::vector<StringRef> &Symbols,
// it's not empty. // it's not empty.
auto TrimmedLine = Line.split('#').first.trim(); auto TrimmedLine = Line.split('#').first.trim();
if (!TrimmedLine.empty()) if (!TrimmedLine.empty())
Symbols.push_back(Saver.save(TrimmedLine)); Symbols.emplace_back(Saver.save(TrimmedLine), UseRegex);
} }
} }
NameOrRegex::NameOrRegex(StringRef Pattern, bool IsRegex) {
if (!IsRegex) {
Name = Pattern;
return;
}
SmallVector<char, 32> Data;
R = std::make_shared<Regex>(
("^" + Pattern.ltrim('^').rtrim('$') + "$").toStringRef(Data));
}
// ParseObjcopyOptions returns the config and sets the input arguments. If a // ParseObjcopyOptions returns the config and sets the input arguments. If a
// help flag is set then ParseObjcopyOptions will print the help messege and // help flag is set then ParseObjcopyOptions will print the help messege and
// exit. // exit.
@ -292,6 +303,7 @@ DriverConfig parseObjcopyOptions(ArrayRef<const char *> ArgsArr) {
InputArgs.hasArg(OBJCOPY_output_target))) InputArgs.hasArg(OBJCOPY_output_target)))
error("--target cannot be used with --input-target or --output-target"); error("--target cannot be used with --input-target or --output-target");
bool UseRegex = InputArgs.hasArg(OBJCOPY_regex);
if (InputArgs.hasArg(OBJCOPY_target)) { if (InputArgs.hasArg(OBJCOPY_target)) {
Config.InputFormat = InputArgs.getLastArgValue(OBJCOPY_target); Config.InputFormat = InputArgs.getLastArgValue(OBJCOPY_target);
Config.OutputFormat = InputArgs.getLastArgValue(OBJCOPY_target); Config.OutputFormat = InputArgs.getLastArgValue(OBJCOPY_target);
@ -371,11 +383,11 @@ DriverConfig parseObjcopyOptions(ArrayRef<const char *> ArgsArr) {
} }
for (auto Arg : InputArgs.filtered(OBJCOPY_remove_section)) for (auto Arg : InputArgs.filtered(OBJCOPY_remove_section))
Config.ToRemove.push_back(Arg->getValue()); Config.ToRemove.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_keep_section)) for (auto Arg : InputArgs.filtered(OBJCOPY_keep_section))
Config.KeepSection.push_back(Arg->getValue()); Config.KeepSection.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_only_section)) for (auto Arg : InputArgs.filtered(OBJCOPY_only_section))
Config.OnlySection.push_back(Arg->getValue()); Config.OnlySection.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_add_section)) for (auto Arg : InputArgs.filtered(OBJCOPY_add_section))
Config.AddSection.push_back(Arg->getValue()); Config.AddSection.push_back(Arg->getValue());
for (auto Arg : InputArgs.filtered(OBJCOPY_dump_section)) for (auto Arg : InputArgs.filtered(OBJCOPY_dump_section))
@ -400,20 +412,20 @@ DriverConfig parseObjcopyOptions(ArrayRef<const char *> ArgsArr) {
Config.DecompressDebugSections = Config.DecompressDebugSections =
InputArgs.hasArg(OBJCOPY_decompress_debug_sections); InputArgs.hasArg(OBJCOPY_decompress_debug_sections);
for (auto Arg : InputArgs.filtered(OBJCOPY_localize_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_localize_symbol))
Config.SymbolsToLocalize.push_back(Arg->getValue()); Config.SymbolsToLocalize.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_keep_global_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_keep_global_symbol))
Config.SymbolsToKeepGlobal.push_back(Arg->getValue()); Config.SymbolsToKeepGlobal.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_keep_global_symbols)) for (auto Arg : InputArgs.filtered(OBJCOPY_keep_global_symbols))
addGlobalSymbolsFromFile(Config.SymbolsToKeepGlobal, DC.Alloc, addGlobalSymbolsFromFile(Config.SymbolsToKeepGlobal, DC.Alloc,
Arg->getValue()); Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_globalize_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_globalize_symbol))
Config.SymbolsToGlobalize.push_back(Arg->getValue()); Config.SymbolsToGlobalize.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_weaken_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_weaken_symbol))
Config.SymbolsToWeaken.push_back(Arg->getValue()); Config.SymbolsToWeaken.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_strip_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_strip_symbol))
Config.SymbolsToRemove.push_back(Arg->getValue()); Config.SymbolsToRemove.emplace_back(Arg->getValue(), UseRegex);
for (auto Arg : InputArgs.filtered(OBJCOPY_keep_symbol)) for (auto Arg : InputArgs.filtered(OBJCOPY_keep_symbol))
Config.SymbolsToKeep.push_back(Arg->getValue()); Config.SymbolsToKeep.emplace_back(Arg->getValue(), UseRegex);
Config.DeterministicArchives = InputArgs.hasFlag( Config.DeterministicArchives = InputArgs.hasFlag(
OBJCOPY_enable_deterministic_archives, OBJCOPY_enable_deterministic_archives,
@ -472,6 +484,7 @@ DriverConfig parseStripOptions(ArrayRef<const char *> ArgsArr) {
error("Multiple input files cannot be used in combination with -o"); error("Multiple input files cannot be used in combination with -o");
CopyConfig Config; CopyConfig Config;
bool UseRegexp = InputArgs.hasArg(STRIP_regex);
Config.StripDebug = InputArgs.hasArg(STRIP_strip_debug); Config.StripDebug = InputArgs.hasArg(STRIP_strip_debug);
if (InputArgs.hasArg(STRIP_discard_all, STRIP_discard_locals)) if (InputArgs.hasArg(STRIP_discard_all, STRIP_discard_locals))
@ -485,16 +498,16 @@ DriverConfig parseStripOptions(ArrayRef<const char *> ArgsArr) {
Config.KeepFileSymbols = InputArgs.hasArg(STRIP_keep_file_symbols); Config.KeepFileSymbols = InputArgs.hasArg(STRIP_keep_file_symbols);
for (auto Arg : InputArgs.filtered(STRIP_keep_section)) for (auto Arg : InputArgs.filtered(STRIP_keep_section))
Config.KeepSection.push_back(Arg->getValue()); Config.KeepSection.emplace_back(Arg->getValue(), UseRegexp);
for (auto Arg : InputArgs.filtered(STRIP_remove_section)) for (auto Arg : InputArgs.filtered(STRIP_remove_section))
Config.ToRemove.push_back(Arg->getValue()); Config.ToRemove.emplace_back(Arg->getValue(), UseRegexp);
for (auto Arg : InputArgs.filtered(STRIP_strip_symbol)) for (auto Arg : InputArgs.filtered(STRIP_strip_symbol))
Config.SymbolsToRemove.push_back(Arg->getValue()); Config.SymbolsToRemove.emplace_back(Arg->getValue(), UseRegexp);
for (auto Arg : InputArgs.filtered(STRIP_keep_symbol)) for (auto Arg : InputArgs.filtered(STRIP_keep_symbol))
Config.SymbolsToKeep.push_back(Arg->getValue()); Config.SymbolsToKeep.emplace_back(Arg->getValue(), UseRegexp);
if (!Config.StripDebug && !Config.StripUnneeded && if (!Config.StripDebug && !Config.StripUnneeded &&
Config.DiscardMode == DiscardType::None && !Config.StripAllGNU && Config.SymbolsToRemove.empty()) Config.DiscardMode == DiscardType::None && !Config.StripAllGNU && Config.SymbolsToRemove.empty())

View File

@ -15,6 +15,7 @@
#include "llvm/ADT/StringMap.h" #include "llvm/ADT/StringMap.h"
#include "llvm/ADT/StringRef.h" #include "llvm/ADT/StringRef.h"
#include "llvm/Support/Allocator.h" #include "llvm/Support/Allocator.h"
#include "llvm/Support/Regex.h"
// Necessary for llvm::DebugCompressionType::None // Necessary for llvm::DebugCompressionType::None
#include "llvm/Target/TargetOptions.h" #include "llvm/Target/TargetOptions.h"
#include <vector> #include <vector>
@ -48,6 +49,17 @@ enum class DiscardType {
Locals, // --discard-locals (-X) Locals, // --discard-locals (-X)
}; };
class NameOrRegex {
StringRef Name;
// Regex is shared between multiple CopyConfig instances.
std::shared_ptr<Regex> R;
public:
NameOrRegex(StringRef Pattern, bool IsRegex);
bool operator==(StringRef S) const { return R ? R->match(S) : Name == S; }
bool operator!=(StringRef S) const { return !operator==(S); }
};
// Configuration for copying/stripping a single file. // Configuration for copying/stripping a single file.
struct CopyConfig { struct CopyConfig {
// Main input/output options // Main input/output options
@ -73,15 +85,15 @@ struct CopyConfig {
// Repeated options // Repeated options
std::vector<StringRef> AddSection; std::vector<StringRef> AddSection;
std::vector<StringRef> DumpSection; std::vector<StringRef> DumpSection;
std::vector<StringRef> KeepSection; std::vector<NameOrRegex> KeepSection;
std::vector<StringRef> OnlySection; std::vector<NameOrRegex> OnlySection;
std::vector<StringRef> SymbolsToGlobalize; std::vector<NameOrRegex> SymbolsToGlobalize;
std::vector<StringRef> SymbolsToKeep; std::vector<NameOrRegex> SymbolsToKeep;
std::vector<StringRef> SymbolsToLocalize; std::vector<NameOrRegex> SymbolsToLocalize;
std::vector<StringRef> SymbolsToRemove; std::vector<NameOrRegex> SymbolsToRemove;
std::vector<StringRef> SymbolsToWeaken; std::vector<NameOrRegex> SymbolsToWeaken;
std::vector<StringRef> ToRemove; std::vector<NameOrRegex> ToRemove;
std::vector<StringRef> SymbolsToKeepGlobal; std::vector<NameOrRegex> SymbolsToKeepGlobal;
// Map options // Map options
StringMap<SectionRename> SectionsToRename; StringMap<SectionRename> SectionsToRename;

View File

@ -193,3 +193,7 @@ defm build_id_link_output
: Eq<"build-id-link-output", "Hard-link the output to <dir>/xx/xxx<suffix> " : Eq<"build-id-link-output", "Hard-link the output to <dir>/xx/xxx<suffix> "
"name derived from hex build ID">, "name derived from hex build ID">,
MetaVarName<"suffix">; MetaVarName<"suffix">;
def regex
: Flag<["-", "--"], "regex">,
HelpText<"Permit regular expressions in name comparison">;

View File

@ -74,6 +74,10 @@ def discard_all
HelpText<"Remove all local symbols except file and section symbols">; HelpText<"Remove all local symbols except file and section symbols">;
def x : Flag<["-"], "x">, Alias<discard_all>; def x : Flag<["-"], "x">, Alias<discard_all>;
def regex
: Flag<["-", "--"], "regex">,
HelpText<"Permit regular expressions in name comparison">;
def version : Flag<["-", "--"], "version">, def version : Flag<["-", "--"], "version">,
HelpText<"Print the version and exit.">; HelpText<"Print the version and exit.">;
def V : Flag<["-"], "V">, Alias<version>; def V : Flag<["-"], "V">, Alias<version>;