forked from OSchip/llvm-project
Fix tab completion for command arguments containing spaces
If a command argument contains a space then it have to be escaped with backslash signs so the argument parsing logic can parse it properly. This CL fixes the tab completion code for the arguments to create complitions with correctly escaped strings. Differential revision: http://reviews.llvm.org/D12531 llvm-svn: 246639
This commit is contained in:
parent
e6035de11e
commit
89d3f090b2
|
@ -424,6 +424,9 @@ public:
|
||||||
static void
|
static void
|
||||||
ExpandEscapedCharacters (const char *src, std::string &dst);
|
ExpandEscapedCharacters (const char *src, std::string &dst);
|
||||||
|
|
||||||
|
static std::string
|
||||||
|
EscapeLLDBCommandArgument (const std::string& arg, char quote_char);
|
||||||
|
|
||||||
// This one isn't really relevant to Arguments per se, but we're using the Args as a
|
// This one isn't really relevant to Arguments per se, but we're using the Args as a
|
||||||
// general strings container, so...
|
// general strings container, so...
|
||||||
void
|
void
|
||||||
|
|
|
@ -1668,3 +1668,33 @@ Args::ExpandEscapedCharacters (const char *src, std::string &dst)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string
|
||||||
|
Args::EscapeLLDBCommandArgument (const std::string& arg, char quote_char)
|
||||||
|
{
|
||||||
|
const char* chars_to_escape = nullptr;
|
||||||
|
switch (quote_char)
|
||||||
|
{
|
||||||
|
case '\0':
|
||||||
|
chars_to_escape = " \t\\'\"`";
|
||||||
|
break;
|
||||||
|
case '\'':
|
||||||
|
chars_to_escape = "";
|
||||||
|
break;
|
||||||
|
case '"':
|
||||||
|
chars_to_escape = "$\"`\\";
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(false && "Unhandled quote character");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string res;
|
||||||
|
res.reserve(arg.size());
|
||||||
|
for (char c : arg)
|
||||||
|
{
|
||||||
|
if (::strchr(chars_to_escape, c))
|
||||||
|
res.push_back('\\');
|
||||||
|
res.push_back(c);
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -2002,7 +2002,7 @@ CommandInterpreter::HandleCompletion (const char *current_line,
|
||||||
matches);
|
matches);
|
||||||
|
|
||||||
if (num_command_matches <= 0)
|
if (num_command_matches <= 0)
|
||||||
return num_command_matches;
|
return num_command_matches;
|
||||||
|
|
||||||
if (num_args == 0)
|
if (num_args == 0)
|
||||||
{
|
{
|
||||||
|
@ -2021,18 +2021,18 @@ CommandInterpreter::HandleCompletion (const char *current_line,
|
||||||
std::string common_prefix;
|
std::string common_prefix;
|
||||||
matches.LongestCommonPrefix (common_prefix);
|
matches.LongestCommonPrefix (common_prefix);
|
||||||
const size_t partial_name_len = command_partial_str.size();
|
const size_t partial_name_len = command_partial_str.size();
|
||||||
|
common_prefix.erase (0, partial_name_len);
|
||||||
|
|
||||||
// If we matched a unique single command, add a space...
|
// If we matched a unique single command, add a space...
|
||||||
// Only do this if the completer told us this was a complete word, however...
|
// Only do this if the completer told us this was a complete word, however...
|
||||||
if (num_command_matches == 1 && word_complete)
|
if (num_command_matches == 1 && word_complete)
|
||||||
{
|
{
|
||||||
char quote_char = parsed_line.GetArgumentQuoteCharAtIndex(cursor_index);
|
char quote_char = parsed_line.GetArgumentQuoteCharAtIndex(cursor_index);
|
||||||
|
common_prefix = Args::EscapeLLDBCommandArgument(common_prefix, quote_char);
|
||||||
if (quote_char != '\0')
|
if (quote_char != '\0')
|
||||||
common_prefix.push_back(quote_char);
|
common_prefix.push_back(quote_char);
|
||||||
|
|
||||||
common_prefix.push_back(' ');
|
common_prefix.push_back(' ');
|
||||||
}
|
}
|
||||||
common_prefix.erase (0, partial_name_len);
|
|
||||||
matches.InsertStringAtIndex(0, common_prefix.c_str());
|
matches.InsertStringAtIndex(0, common_prefix.c_str());
|
||||||
}
|
}
|
||||||
return num_command_matches;
|
return num_command_matches;
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
LEVEL = ../../make
|
||||||
|
|
||||||
|
CXX_SOURCES := main.cpp
|
||||||
|
|
||||||
|
include $(LEVEL)/Makefile.rules
|
|
@ -219,6 +219,23 @@ class CommandLineCompletionTestCase(TestBase):
|
||||||
"""Test that 'target va' completes to 'target variable '."""
|
"""Test that 'target va' completes to 'target variable '."""
|
||||||
self.complete_from_to('target va', 'target variable ')
|
self.complete_from_to('target va', 'target variable ')
|
||||||
|
|
||||||
|
@skipUnlessDarwin
|
||||||
|
@dsym_test
|
||||||
|
def test_symbol_name_dsym(self):
|
||||||
|
self.buildDsym()
|
||||||
|
self.complete_from_to('''file a.out
|
||||||
|
breakpoint set -n Fo''',
|
||||||
|
'breakpoint set -n Foo::Bar(int,\\ int)',
|
||||||
|
turn_off_re_match=True)
|
||||||
|
|
||||||
|
@dwarf_test
|
||||||
|
def test_symbol_name_dwarf(self):
|
||||||
|
self.buildDwarf()
|
||||||
|
self.complete_from_to('''file a.out
|
||||||
|
breakpoint set -n Fo''',
|
||||||
|
'breakpoint set -n Foo::Bar(int,\\ int)',
|
||||||
|
turn_off_re_match=True)
|
||||||
|
|
||||||
def complete_from_to(self, str_input, patterns, turn_off_re_match=False):
|
def complete_from_to(self, str_input, patterns, turn_off_re_match=False):
|
||||||
"""Test that the completion mechanism completes str_input to patterns,
|
"""Test that the completion mechanism completes str_input to patterns,
|
||||||
where patterns could be a pattern-string or a list of pattern-strings"""
|
where patterns could be a pattern-string or a list of pattern-strings"""
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
class Foo
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
int Bar(int x, int y)
|
||||||
|
{
|
||||||
|
return x + y;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
Foo f;
|
||||||
|
f.Bar(1, 2);
|
||||||
|
}
|
Loading…
Reference in New Issue