|  | //===-- OptionGroupFormat.cpp ---------------------------------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "lldb/Interpreter/OptionGroupFormat.h" | 
|  |  | 
|  | #include "lldb/Host/OptionParser.h" | 
|  | #include "lldb/Interpreter/CommandInterpreter.h" | 
|  | #include "lldb/Target/ExecutionContext.h" | 
|  | #include "lldb/Target/Target.h" | 
|  |  | 
|  | using namespace lldb; | 
|  | using namespace lldb_private; | 
|  |  | 
|  | OptionGroupFormat::OptionGroupFormat(lldb::Format default_format, | 
|  | uint64_t default_byte_size, | 
|  | uint64_t default_count) | 
|  | : m_format(default_format, default_format), | 
|  | m_byte_size(default_byte_size, default_byte_size), | 
|  | m_count(default_count, default_count), m_prev_gdb_format('x'), | 
|  | m_prev_gdb_size('w') {} | 
|  |  | 
|  | OptionGroupFormat::~OptionGroupFormat() {} | 
|  |  | 
|  | static constexpr OptionDefinition g_option_table[] = { | 
|  | {LLDB_OPT_SET_1, false, "format", 'f', OptionParser::eRequiredArgument, | 
|  | nullptr, {}, 0, eArgTypeFormat, | 
|  | "Specify a format to be used for display."}, | 
|  | {LLDB_OPT_SET_2, false, "gdb-format", 'G', OptionParser::eRequiredArgument, | 
|  | nullptr, {}, 0, eArgTypeGDBFormat, | 
|  | "Specify a format using a GDB format specifier string."}, | 
|  | {LLDB_OPT_SET_3, false, "size", 's', OptionParser::eRequiredArgument, | 
|  | nullptr, {}, 0, eArgTypeByteSize, | 
|  | "The size in bytes to use when displaying with the selected format."}, | 
|  | {LLDB_OPT_SET_4, false, "count", 'c', OptionParser::eRequiredArgument, | 
|  | nullptr, {}, 0, eArgTypeCount, | 
|  | "The number of total items to display."}, | 
|  | }; | 
|  |  | 
|  | llvm::ArrayRef<OptionDefinition> OptionGroupFormat::GetDefinitions() { | 
|  | auto result = llvm::makeArrayRef(g_option_table); | 
|  | if (m_byte_size.GetDefaultValue() < UINT64_MAX) { | 
|  | if (m_count.GetDefaultValue() < UINT64_MAX) | 
|  | return result; | 
|  | else | 
|  | return result.take_front(3); | 
|  | } | 
|  | return result.take_front(2); | 
|  | } | 
|  |  | 
|  | Status OptionGroupFormat::SetOptionValue(uint32_t option_idx, | 
|  | llvm::StringRef option_arg, | 
|  | ExecutionContext *execution_context) { | 
|  | Status error; | 
|  | const int short_option = g_option_table[option_idx].short_option; | 
|  |  | 
|  | switch (short_option) { | 
|  | case 'f': | 
|  | error = m_format.SetValueFromString(option_arg); | 
|  | break; | 
|  |  | 
|  | case 'c': | 
|  | if (m_count.GetDefaultValue() == 0) { | 
|  | error.SetErrorString("--count option is disabled"); | 
|  | } else { | 
|  | error = m_count.SetValueFromString(option_arg); | 
|  | if (m_count.GetCurrentValue() == 0) | 
|  | error.SetErrorStringWithFormat("invalid --count option value '%s'", | 
|  | option_arg.str().c_str()); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case 's': | 
|  | if (m_byte_size.GetDefaultValue() == 0) { | 
|  | error.SetErrorString("--size option is disabled"); | 
|  | } else { | 
|  | error = m_byte_size.SetValueFromString(option_arg); | 
|  | if (m_byte_size.GetCurrentValue() == 0) | 
|  | error.SetErrorStringWithFormat("invalid --size option value '%s'", | 
|  | option_arg.str().c_str()); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case 'G': { | 
|  | uint64_t count = 0; | 
|  | llvm::StringRef gdb_format_str = option_arg; | 
|  | gdb_format_str.consumeInteger(0, count); | 
|  |  | 
|  | Format format = eFormatDefault; | 
|  | uint32_t byte_size = 0; | 
|  |  | 
|  | while (!gdb_format_str.empty() && | 
|  | ParserGDBFormatLetter(execution_context, gdb_format_str[0], format, | 
|  | byte_size)) { | 
|  | gdb_format_str = gdb_format_str.drop_front(); | 
|  | } | 
|  |  | 
|  | // We the first character of the "gdb_format_str" is not the | 
|  | // NULL terminator, we didn't consume the entire string and | 
|  | // something is wrong. Also, if none of the format, size or count was | 
|  | // specified correctly, then abort. | 
|  | if (!gdb_format_str.empty() || | 
|  | (format == eFormatInvalid && byte_size == 0 && count == 0)) { | 
|  | // Nothing got set correctly | 
|  | error.SetErrorStringWithFormat("invalid gdb format string '%s'", | 
|  | option_arg.str().c_str()); | 
|  | return error; | 
|  | } | 
|  |  | 
|  | // At least one of the format, size or count was set correctly. Anything | 
|  | // that wasn't set correctly should be set to the previous default | 
|  | if (format == eFormatInvalid) | 
|  | ParserGDBFormatLetter(execution_context, m_prev_gdb_format, format, | 
|  | byte_size); | 
|  |  | 
|  | const bool byte_size_enabled = m_byte_size.GetDefaultValue() < UINT64_MAX; | 
|  | const bool count_enabled = m_count.GetDefaultValue() < UINT64_MAX; | 
|  | if (byte_size_enabled) { | 
|  | // Byte size is enabled | 
|  | if (byte_size == 0) | 
|  | ParserGDBFormatLetter(execution_context, m_prev_gdb_size, format, | 
|  | byte_size); | 
|  | } else { | 
|  | // Byte size is disabled, make sure it wasn't specified but if this is an | 
|  | // address, it's actually necessary to specify one so don't error out | 
|  | if (byte_size > 0 && format != lldb::eFormatAddressInfo) { | 
|  | error.SetErrorString( | 
|  | "this command doesn't support specifying a byte size"); | 
|  | return error; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (count_enabled) { | 
|  | // Count is enabled and was not set, set it to the default for gdb format | 
|  | // statements (which is 1). | 
|  | if (count == 0) | 
|  | count = 1; | 
|  | } else { | 
|  | // Count is disabled, make sure it wasn't specified | 
|  | if (count > 0) { | 
|  | error.SetErrorString("this command doesn't support specifying a count"); | 
|  | return error; | 
|  | } | 
|  | } | 
|  |  | 
|  | m_format.SetCurrentValue(format); | 
|  | m_format.SetOptionWasSet(); | 
|  | if (byte_size_enabled) { | 
|  | m_byte_size.SetCurrentValue(byte_size); | 
|  | m_byte_size.SetOptionWasSet(); | 
|  | } | 
|  | if (count_enabled) { | 
|  | m_count.SetCurrentValue(count); | 
|  | m_count.SetOptionWasSet(); | 
|  | } | 
|  | } break; | 
|  |  | 
|  | default: | 
|  | llvm_unreachable("Unimplemented option"); | 
|  | } | 
|  |  | 
|  | return error; | 
|  | } | 
|  |  | 
|  | bool OptionGroupFormat::ParserGDBFormatLetter( | 
|  | ExecutionContext *execution_context, char format_letter, Format &format, | 
|  | uint32_t &byte_size) { | 
|  | m_has_gdb_format = true; | 
|  | switch (format_letter) { | 
|  | case 'o': | 
|  | format = eFormatOctal; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'x': | 
|  | format = eFormatHex; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'd': | 
|  | format = eFormatDecimal; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'u': | 
|  | format = eFormatUnsigned; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 't': | 
|  | format = eFormatBinary; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'f': | 
|  | format = eFormatFloat; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'a': | 
|  | format = eFormatAddressInfo; | 
|  | { | 
|  | TargetSP target_sp = | 
|  | execution_context ? execution_context->GetTargetSP() : TargetSP(); | 
|  | if (target_sp) | 
|  | byte_size = target_sp->GetArchitecture().GetAddressByteSize(); | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | } | 
|  | case 'i': | 
|  | format = eFormatInstruction; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'c': | 
|  | format = eFormatChar; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 's': | 
|  | format = eFormatCString; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'T': | 
|  | format = eFormatOSType; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  | case 'A': | 
|  | format = eFormatHexFloat; | 
|  | m_prev_gdb_format = format_letter; | 
|  | return true; | 
|  |  | 
|  | case 'b': | 
|  | case 'h': | 
|  | case 'w': | 
|  | case 'g': | 
|  | { | 
|  | // Size isn't used for printing instructions, so if a size is specified, | 
|  | // and the previous format was 'i', then we should reset it to the | 
|  | // default ('x').  Otherwise we'll continue to print as instructions, | 
|  | // which isn't expected. | 
|  | if (format_letter == 'b') | 
|  | byte_size = 1; | 
|  | else if (format_letter == 'h') | 
|  | byte_size = 2; | 
|  | else if (format_letter == 'w') | 
|  | byte_size = 4; | 
|  | else if (format_letter == 'g') | 
|  | byte_size = 8; | 
|  |  | 
|  | m_prev_gdb_size = format_letter; | 
|  | if (m_prev_gdb_format == 'i') | 
|  | m_prev_gdb_format = 'x'; | 
|  | return true; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  |  | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void OptionGroupFormat::OptionParsingStarting( | 
|  | ExecutionContext *execution_context) { | 
|  | m_format.Clear(); | 
|  | m_byte_size.Clear(); | 
|  | m_count.Clear(); | 
|  | m_has_gdb_format = false; | 
|  | } |