diff options
Diffstat (limited to 'contrib/llvm/tools/lldb/source/Interpreter/Args.cpp')
-rw-r--r-- | contrib/llvm/tools/lldb/source/Interpreter/Args.cpp | 463 |
1 files changed, 160 insertions, 303 deletions
diff --git a/contrib/llvm/tools/lldb/source/Interpreter/Args.cpp b/contrib/llvm/tools/lldb/source/Interpreter/Args.cpp index 9393380..2258c26 100644 --- a/contrib/llvm/tools/lldb/source/Interpreter/Args.cpp +++ b/contrib/llvm/tools/lldb/source/Interpreter/Args.cpp @@ -7,8 +7,6 @@ // //===----------------------------------------------------------------------===// -#include "lldb/lldb-python.h" - // C Includes #include <cstdlib> // C++ Includes @@ -19,14 +17,13 @@ #include "lldb/Core/StreamFile.h" #include "lldb/Core/StreamString.h" #include "lldb/DataFormatters/FormatManager.h" +#include "lldb/Host/StringConvert.h" #include "lldb/Interpreter/Options.h" #include "lldb/Interpreter/CommandInterpreter.h" #include "lldb/Interpreter/CommandReturnObject.h" #include "lldb/Target/Process.h" -//#include "lldb/Target/RegisterContext.h" #include "lldb/Target/StackFrame.h" #include "lldb/Target/Target.h" -//#include "lldb/Target/Thread.h" using namespace lldb; using namespace lldb_private; @@ -34,25 +31,15 @@ using namespace lldb_private; //---------------------------------------------------------------------- // Args constructor //---------------------------------------------------------------------- -Args::Args (const char *command) : +Args::Args (llvm::StringRef command) : m_args(), m_argv(), m_args_quote_char() { - if (command) - SetCommandString (command); + SetCommandString (command); } -Args::Args (const char *command, size_t len) : - m_args(), - m_argv(), - m_args_quote_char() -{ - if (command && len) - SetCommandString (command, len); -} - //---------------------------------------------------------------------- // We have to be very careful on the copy constructor of this class // to make sure we copy all of the string values, but we can't copy the @@ -147,216 +134,158 @@ Args::GetQuotedCommandString (std::string &command) const return argc > 0; } -void -Args::SetCommandString (const char *command, size_t len) +// A helper function for argument parsing. +// Parses the initial part of the first argument using normal double quote rules: +// backslash escapes the double quote and itself. The parsed string is appended to the second +// argument. The function returns the unparsed portion of the string, starting at the closing +// quote. +static llvm::StringRef +ParseDoubleQuotes(llvm::StringRef quoted, std::string &result) { - // Use std::string to make sure we get a NULL terminated string we can use - // as "command" could point to a string within a large string.... - std::string null_terminated_command(command, len); - SetCommandString(null_terminated_command.c_str()); + // Inside double quotes, '\' and '"' are special. + static const char *k_escapable_characters = "\"\\"; + while (true) + { + // Skip over over regular characters and append them. + size_t regular = quoted.find_first_of(k_escapable_characters); + result += quoted.substr(0, regular); + quoted = quoted.substr(regular); + + // If we have reached the end of string or the closing quote, we're done. + if (quoted.empty() || quoted.front() == '"') + break; + + // We have found a backslash. + quoted = quoted.drop_front(); + + if (quoted.empty()) + { + // A lone backslash at the end of string, let's just append it. + result += '\\'; + break; + } + + // If the character after the backslash is not a whitelisted escapable character, we + // leave the character sequence untouched. + if (strchr(k_escapable_characters, quoted.front()) == nullptr) + result += '\\'; + + result += quoted.front(); + quoted = quoted.drop_front(); + } + + return quoted; } -void -Args::SetCommandString (const char *command) +// A helper function for SetCommandString. +// Parses a single argument from the command string, processing quotes and backslashes in a +// shell-like manner. The parsed argument is appended to the m_args array. The function returns +// the unparsed portion of the string, starting at the first unqouted, unescaped whitespace +// character. +llvm::StringRef +Args::ParseSingleArgument(llvm::StringRef command) { - m_args.clear(); - m_argv.clear(); - m_args_quote_char.clear(); - - if (command && command[0]) + // Argument can be split into multiple discontiguous pieces, + // for example: + // "Hello ""World" + // this would result in a single argument "Hello World" (without/ + // the quotes) since the quotes would be removed and there is + // not space between the strings. + + std::string arg; + + // Since we can have multiple quotes that form a single command + // in a command like: "Hello "world'!' (which will make a single + // argument "Hello world!") we remember the first quote character + // we encounter and use that for the quote character. + char first_quote_char = '\0'; + + bool arg_complete = false; + do { - static const char *k_space_separators = " \t"; - static const char *k_space_separators_with_slash_and_quotes = " \t \\'\""; - const char *arg_end = nullptr; - const char *arg_pos; - for (arg_pos = command; - arg_pos && arg_pos[0]; - arg_pos = arg_end) + // Skip over over regular characters and append them. + size_t regular = command.find_first_of(" \t\"'`\\"); + arg += command.substr(0, regular); + command = command.substr(regular); + + if (command.empty()) + break; + + char special = command.front(); + command = command.drop_front(); + switch (special) { - // Skip any leading space separators - const char *arg_start = ::strspn (arg_pos, k_space_separators) + arg_pos; - - // If there were only space separators to the end of the line, then - // we're done. - if (*arg_start == '\0') + case '\\': + if (command.empty()) + { + arg += '\\'; break; + } - // Arguments can be split into multiple discontiguous pieces, - // for example: - // "Hello ""World" - // this would result in a single argument "Hello World" (without/ - // the quotes) since the quotes would be removed and there is - // not space between the strings. So we need to keep track of the - // current start of each argument piece in "arg_piece_start" - const char *arg_piece_start = arg_start; - arg_pos = arg_piece_start; - - std::string arg; - // Since we can have multiple quotes that form a single command - // in a command like: "Hello "world'!' (which will make a single - // argument "Hello world!") we remember the first quote character - // we encounter and use that for the quote character. - char first_quote_char = '\0'; - char quote_char = '\0'; - bool arg_complete = false; - - do - { - arg_end = ::strcspn (arg_pos, k_space_separators_with_slash_and_quotes) + arg_pos; + // If the character after the backslash is not a whitelisted escapable character, we + // leave the character sequence untouched. + if (strchr(" \t\\'\"`", command.front()) == nullptr) + arg += '\\'; - switch (arg_end[0]) - { - default: - assert (!"Unhandled case statement, we must handle this..."); - break; + arg += command.front(); + command = command.drop_front(); - case '\0': - // End of C string - if (arg_piece_start && arg_piece_start[0]) - arg.append (arg_piece_start); - arg_complete = true; - break; - - case '\\': - // Backslash character - switch (arg_end[1]) - { - case '\0': - arg.append (arg_piece_start); - ++arg_end; - arg_complete = true; - break; - - default: - if (quote_char == '\0') - { - arg.append (arg_piece_start, arg_end - arg_piece_start); - if (arg_end[1] != '\0') - { - arg.append (arg_end + 1, 1); - arg_pos = arg_end + 2; - arg_piece_start = arg_pos; - } - } - else - arg_pos = arg_end + 2; - break; - } - break; - - case '"': - case '\'': - case '`': - // Quote characters - if (quote_char) - { - // We found a quote character while inside a quoted - // character argument. If it matches our current quote - // character, this ends the effect of the quotes. If it - // doesn't we ignore it. - if (quote_char == arg_end[0]) - { - arg.append (arg_piece_start, arg_end - arg_piece_start); - // Clear the quote character and let parsing - // continue (we need to watch for things like: - // "Hello ""World" - // "Hello "World - // "Hello "'World' - // All of which will result in a single argument "Hello World" - quote_char = '\0'; // Note that we are no longer inside quotes - arg_pos = arg_end + 1; // Skip the quote character - arg_piece_start = arg_pos; // Note we are starting from later in the string - } - else - { - // different quote, skip it and keep going - arg_pos = arg_end + 1; - } - } - else - { - // We found the start of a quote scope. - // Make sure there isn't a string that precedes - // the start of a quote scope like: - // Hello" World" - // If so, then add the "Hello" to the arg - if (arg_end > arg_piece_start) - arg.append (arg_piece_start, arg_end - arg_piece_start); - - // Enter into a quote scope - quote_char = arg_end[0]; - - if (first_quote_char == '\0') - first_quote_char = quote_char; + break; - arg_pos = arg_end; - ++arg_pos; // Skip the quote character - arg_piece_start = arg_pos; // Note we are starting from later in the string - - // Skip till the next quote character - const char *end_quote = ::strchr (arg_piece_start, quote_char); - while (end_quote && end_quote[-1] == '\\') - { - // Don't skip the quote character if it is - // preceded by a '\' character - end_quote = ::strchr (end_quote + 1, quote_char); - } - - if (end_quote) - { - if (end_quote > arg_piece_start) - arg.append (arg_piece_start, end_quote - arg_piece_start); + case ' ': + case '\t': + // We are not inside any quotes, we just found a space after an + // argument. We are done. + arg_complete = true; + break; - // If the next character is a space or the end of - // string, this argument is complete... - if (end_quote[1] == ' ' || end_quote[1] == '\t' || end_quote[1] == '\0') - { - arg_complete = true; - arg_end = end_quote + 1; - } - else - { - arg_pos = end_quote + 1; - arg_piece_start = arg_pos; - } - quote_char = '\0'; - } - else - { - // Consume the rest of the string as there was no terminating quote - arg.append(arg_piece_start); - arg_end = arg_piece_start + strlen(arg_piece_start); - arg_complete = true; - } - } - break; + case '"': + case '\'': + case '`': + // We found the start of a quote scope. + if (first_quote_char == '\0') + first_quote_char = special; - case ' ': - case '\t': - if (quote_char) - { - // We are currently processing a quoted character and found - // a space character, skip any spaces and keep trying to find - // the end of the argument. - arg_pos = ::strspn (arg_end, k_space_separators) + arg_end; - } - else - { - // We are not inside any quotes, we just found a space after an - // argument - if (arg_end > arg_piece_start) - arg.append (arg_piece_start, arg_end - arg_piece_start); - arg_complete = true; - } - break; - } - } while (!arg_complete); + if (special == '"') + command = ParseDoubleQuotes(command, arg); + else + { + // For single quotes, we simply skip ahead to the matching quote character + // (or the end of the string). + size_t quoted = command.find(special); + arg += command.substr(0, quoted); + command = command.substr(quoted); + } + + // If we found a closing quote, skip it. + if (! command.empty()) + command = command.drop_front(); - m_args.push_back(arg); - m_args_quote_char.push_back (first_quote_char); + break; } - UpdateArgvFromArgs(); + } while (!arg_complete); + + m_args.push_back(arg); + m_args_quote_char.push_back (first_quote_char); + return command; +} + +void +Args::SetCommandString (llvm::StringRef command) +{ + m_args.clear(); + m_argv.clear(); + m_args_quote_char.clear(); + + static const char *k_space_separators = " \t"; + command = command.ltrim(k_space_separators); + while (!command.empty()) + { + command = ParseSingleArgument(command); + command = command.ltrim(k_space_separators); } + + UpdateArgvFromArgs(); } void @@ -481,7 +410,8 @@ Args::AppendArguments (const Args &rhs) { const size_t rhs_argc = rhs.GetArgumentCount(); for (size_t i=0; i<rhs_argc; ++i) - AppendArgument(rhs.GetArgumentAtIndex(i)); + AppendArgument(rhs.GetArgumentAtIndex(i), + rhs.GetArgumentQuoteCharAtIndex(i)); } void @@ -722,77 +652,6 @@ Args::Clear () m_args_quote_char.clear(); } -int32_t -Args::StringToSInt32 (const char *s, int32_t fail_value, int base, bool *success_ptr) -{ - if (s && s[0]) - { - char *end = nullptr; - const long sval = ::strtol (s, &end, base); - if (*end == '\0') - { - if (success_ptr) - *success_ptr = ((sval <= INT32_MAX) && (sval >= INT32_MIN)); - return (int32_t)sval; // All characters were used, return the result - } - } - if (success_ptr) *success_ptr = false; - return fail_value; -} - -uint32_t -Args::StringToUInt32 (const char *s, uint32_t fail_value, int base, bool *success_ptr) -{ - if (s && s[0]) - { - char *end = nullptr; - const unsigned long uval = ::strtoul (s, &end, base); - if (*end == '\0') - { - if (success_ptr) - *success_ptr = (uval <= UINT32_MAX); - return (uint32_t)uval; // All characters were used, return the result - } - } - if (success_ptr) *success_ptr = false; - return fail_value; -} - - -int64_t -Args::StringToSInt64 (const char *s, int64_t fail_value, int base, bool *success_ptr) -{ - if (s && s[0]) - { - char *end = nullptr; - int64_t uval = ::strtoll (s, &end, base); - if (*end == '\0') - { - if (success_ptr) *success_ptr = true; - return uval; // All characters were used, return the result - } - } - if (success_ptr) *success_ptr = false; - return fail_value; -} - -uint64_t -Args::StringToUInt64 (const char *s, uint64_t fail_value, int base, bool *success_ptr) -{ - if (s && s[0]) - { - char *end = nullptr; - uint64_t uval = ::strtoull (s, &end, base); - if (*end == '\0') - { - if (success_ptr) *success_ptr = true; - return uval; // All characters were used, return the result - } - } - if (success_ptr) *success_ptr = false; - return fail_value; -} - lldb::addr_t Args::StringToAddress (const ExecutionContext *exe_ctx, const char *s, lldb::addr_t fail_value, Error *error_ptr) { @@ -878,7 +737,7 @@ Args::StringToAddress (const ExecutionContext *exe_ctx, const char *s, lldb::add if (regex_match.GetMatchAtIndex(s, 3, str)) { - offset = Args::StringToUInt64(str.c_str(), 0, 0, &success); + offset = StringConvert::ToUInt64(str.c_str(), 0, 0, &success); if (success) { @@ -944,26 +803,24 @@ Args::StripSpaces (std::string &s, bool leading, bool trailing, bool return_null bool Args::StringToBoolean (const char *s, bool fail_value, bool *success_ptr) { - if (s && s[0]) + llvm::StringRef ref = llvm::StringRef(s).trim(); + if (ref.equals_lower("false") || + ref.equals_lower("off") || + ref.equals_lower("no") || + ref.equals_lower("0")) { - if (::strcasecmp (s, "false") == 0 || - ::strcasecmp (s, "off") == 0 || - ::strcasecmp (s, "no") == 0 || - ::strcmp (s, "0") == 0) - { - if (success_ptr) - *success_ptr = true; - return false; - } - else - if (::strcasecmp (s, "true") == 0 || - ::strcasecmp (s, "on") == 0 || - ::strcasecmp (s, "yes") == 0 || - ::strcmp (s, "1") == 0) - { - if (success_ptr) *success_ptr = true; - return true; - } + if (success_ptr) + *success_ptr = true; + return false; + } + else + if (ref.equals_lower("true") || + ref.equals_lower("on") || + ref.equals_lower("yes") || + ref.equals_lower("1")) + { + if (success_ptr) *success_ptr = true; + return true; } if (success_ptr) *success_ptr = false; return fail_value; @@ -1279,7 +1136,7 @@ Args::IsPositionalArgument (const char *arg) return false; bool is_positional = true; - char *cptr = (char *) arg; + const char *cptr = arg; if (cptr[0] == '%') { |