xref: /freebsd-src/contrib/llvm-project/lldb/source/Interpreter/OptionGroupFormat.cpp (revision bdd1243df58e60e85101c09001d9812a789b6bc4)
15ffd83dbSDimitry Andric //===-- OptionGroupFormat.cpp ---------------------------------------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric 
90b57cec5SDimitry Andric #include "lldb/Interpreter/OptionGroupFormat.h"
100b57cec5SDimitry Andric 
110b57cec5SDimitry Andric #include "lldb/Host/OptionParser.h"
120b57cec5SDimitry Andric #include "lldb/Interpreter/CommandInterpreter.h"
130b57cec5SDimitry Andric #include "lldb/Target/ExecutionContext.h"
140b57cec5SDimitry Andric #include "lldb/Target/Target.h"
150b57cec5SDimitry Andric 
160b57cec5SDimitry Andric using namespace lldb;
170b57cec5SDimitry Andric using namespace lldb_private;
180b57cec5SDimitry Andric 
194824e7fdSDimitry Andric static constexpr OptionDefinition g_default_option_definitions[] = {
200b57cec5SDimitry Andric     {LLDB_OPT_SET_1, false, "format", 'f', OptionParser::eRequiredArgument,
210b57cec5SDimitry Andric      nullptr, {}, 0, eArgTypeFormat,
220b57cec5SDimitry Andric      "Specify a format to be used for display."},
230b57cec5SDimitry Andric     {LLDB_OPT_SET_2, false, "gdb-format", 'G', OptionParser::eRequiredArgument,
240b57cec5SDimitry Andric      nullptr, {}, 0, eArgTypeGDBFormat,
250b57cec5SDimitry Andric      "Specify a format using a GDB format specifier string."},
260b57cec5SDimitry Andric     {LLDB_OPT_SET_3, false, "size", 's', OptionParser::eRequiredArgument,
270b57cec5SDimitry Andric      nullptr, {}, 0, eArgTypeByteSize,
280b57cec5SDimitry Andric      "The size in bytes to use when displaying with the selected format."},
290b57cec5SDimitry Andric     {LLDB_OPT_SET_4, false, "count", 'c', OptionParser::eRequiredArgument,
300b57cec5SDimitry Andric      nullptr, {}, 0, eArgTypeCount,
310b57cec5SDimitry Andric      "The number of total items to display."},
320b57cec5SDimitry Andric };
330b57cec5SDimitry Andric 
OptionGroupFormat(lldb::Format default_format,uint64_t default_byte_size,uint64_t default_count,OptionGroupFormatUsageTextVector usage_text_vector)344824e7fdSDimitry Andric OptionGroupFormat::OptionGroupFormat(
354824e7fdSDimitry Andric     lldb::Format default_format, uint64_t default_byte_size,
364824e7fdSDimitry Andric     uint64_t default_count, OptionGroupFormatUsageTextVector usage_text_vector)
374824e7fdSDimitry Andric     : m_format(default_format, default_format),
384824e7fdSDimitry Andric       m_byte_size(default_byte_size, default_byte_size),
394824e7fdSDimitry Andric       m_count(default_count, default_count), m_prev_gdb_format('x'),
40*bdd1243dSDimitry Andric       m_prev_gdb_size('w'), m_has_gdb_format(false) {
414824e7fdSDimitry Andric   // Copy the default option definitions.
424824e7fdSDimitry Andric   std::copy(std::begin(g_default_option_definitions),
434824e7fdSDimitry Andric             std::end(g_default_option_definitions),
444824e7fdSDimitry Andric             std::begin(m_option_definitions));
454824e7fdSDimitry Andric 
464824e7fdSDimitry Andric   for (auto usage_text_tuple : usage_text_vector) {
474824e7fdSDimitry Andric     switch (std::get<0>(usage_text_tuple)) {
484824e7fdSDimitry Andric     case eArgTypeFormat:
494824e7fdSDimitry Andric       m_option_definitions[0].usage_text = std::get<1>(usage_text_tuple);
504824e7fdSDimitry Andric       break;
514824e7fdSDimitry Andric     case eArgTypeByteSize:
524824e7fdSDimitry Andric       m_option_definitions[2].usage_text = std::get<1>(usage_text_tuple);
534824e7fdSDimitry Andric       break;
544824e7fdSDimitry Andric     default:
554824e7fdSDimitry Andric       llvm_unreachable("Unimplemented option");
564824e7fdSDimitry Andric     }
574824e7fdSDimitry Andric   }
584824e7fdSDimitry Andric }
594824e7fdSDimitry Andric 
GetDefinitions()600b57cec5SDimitry Andric llvm::ArrayRef<OptionDefinition> OptionGroupFormat::GetDefinitions() {
61*bdd1243dSDimitry Andric   auto result = llvm::ArrayRef(m_option_definitions);
620b57cec5SDimitry Andric   if (m_byte_size.GetDefaultValue() < UINT64_MAX) {
630b57cec5SDimitry Andric     if (m_count.GetDefaultValue() < UINT64_MAX)
640b57cec5SDimitry Andric       return result;
650b57cec5SDimitry Andric     else
660b57cec5SDimitry Andric       return result.take_front(3);
670b57cec5SDimitry Andric   }
680b57cec5SDimitry Andric   return result.take_front(2);
690b57cec5SDimitry Andric }
700b57cec5SDimitry Andric 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)710b57cec5SDimitry Andric Status OptionGroupFormat::SetOptionValue(uint32_t option_idx,
720b57cec5SDimitry Andric                                          llvm::StringRef option_arg,
730b57cec5SDimitry Andric                                          ExecutionContext *execution_context) {
740b57cec5SDimitry Andric   Status error;
754824e7fdSDimitry Andric   const int short_option = m_option_definitions[option_idx].short_option;
760b57cec5SDimitry Andric 
770b57cec5SDimitry Andric   switch (short_option) {
780b57cec5SDimitry Andric   case 'f':
790b57cec5SDimitry Andric     error = m_format.SetValueFromString(option_arg);
800b57cec5SDimitry Andric     break;
810b57cec5SDimitry Andric 
820b57cec5SDimitry Andric   case 'c':
830b57cec5SDimitry Andric     if (m_count.GetDefaultValue() == 0) {
840b57cec5SDimitry Andric       error.SetErrorString("--count option is disabled");
850b57cec5SDimitry Andric     } else {
860b57cec5SDimitry Andric       error = m_count.SetValueFromString(option_arg);
870b57cec5SDimitry Andric       if (m_count.GetCurrentValue() == 0)
880b57cec5SDimitry Andric         error.SetErrorStringWithFormat("invalid --count option value '%s'",
890b57cec5SDimitry Andric                                        option_arg.str().c_str());
900b57cec5SDimitry Andric     }
910b57cec5SDimitry Andric     break;
920b57cec5SDimitry Andric 
930b57cec5SDimitry Andric   case 's':
940b57cec5SDimitry Andric     if (m_byte_size.GetDefaultValue() == 0) {
950b57cec5SDimitry Andric       error.SetErrorString("--size option is disabled");
960b57cec5SDimitry Andric     } else {
970b57cec5SDimitry Andric       error = m_byte_size.SetValueFromString(option_arg);
980b57cec5SDimitry Andric       if (m_byte_size.GetCurrentValue() == 0)
990b57cec5SDimitry Andric         error.SetErrorStringWithFormat("invalid --size option value '%s'",
1000b57cec5SDimitry Andric                                        option_arg.str().c_str());
1010b57cec5SDimitry Andric     }
1020b57cec5SDimitry Andric     break;
1030b57cec5SDimitry Andric 
1040b57cec5SDimitry Andric   case 'G': {
1050b57cec5SDimitry Andric     uint64_t count = 0;
1060b57cec5SDimitry Andric     llvm::StringRef gdb_format_str = option_arg;
1070b57cec5SDimitry Andric     gdb_format_str.consumeInteger(0, count);
1080b57cec5SDimitry Andric 
1090b57cec5SDimitry Andric     Format format = eFormatDefault;
1100b57cec5SDimitry Andric     uint32_t byte_size = 0;
1110b57cec5SDimitry Andric 
1120b57cec5SDimitry Andric     while (!gdb_format_str.empty() &&
1130b57cec5SDimitry Andric            ParserGDBFormatLetter(execution_context, gdb_format_str[0], format,
1140b57cec5SDimitry Andric                                  byte_size)) {
1150b57cec5SDimitry Andric       gdb_format_str = gdb_format_str.drop_front();
1160b57cec5SDimitry Andric     }
1170b57cec5SDimitry Andric 
1180b57cec5SDimitry Andric     // We the first character of the "gdb_format_str" is not the
1190b57cec5SDimitry Andric     // NULL terminator, we didn't consume the entire string and
1200b57cec5SDimitry Andric     // something is wrong. Also, if none of the format, size or count was
1210b57cec5SDimitry Andric     // specified correctly, then abort.
1220b57cec5SDimitry Andric     if (!gdb_format_str.empty() ||
1230b57cec5SDimitry Andric         (format == eFormatInvalid && byte_size == 0 && count == 0)) {
1240b57cec5SDimitry Andric       // Nothing got set correctly
1250b57cec5SDimitry Andric       error.SetErrorStringWithFormat("invalid gdb format string '%s'",
1260b57cec5SDimitry Andric                                      option_arg.str().c_str());
1270b57cec5SDimitry Andric       return error;
1280b57cec5SDimitry Andric     }
1290b57cec5SDimitry Andric 
1300b57cec5SDimitry Andric     // At least one of the format, size or count was set correctly. Anything
1310b57cec5SDimitry Andric     // that wasn't set correctly should be set to the previous default
1320b57cec5SDimitry Andric     if (format == eFormatInvalid)
1330b57cec5SDimitry Andric       ParserGDBFormatLetter(execution_context, m_prev_gdb_format, format,
1340b57cec5SDimitry Andric                             byte_size);
1350b57cec5SDimitry Andric 
1360b57cec5SDimitry Andric     const bool byte_size_enabled = m_byte_size.GetDefaultValue() < UINT64_MAX;
1370b57cec5SDimitry Andric     const bool count_enabled = m_count.GetDefaultValue() < UINT64_MAX;
1380b57cec5SDimitry Andric     if (byte_size_enabled) {
1390b57cec5SDimitry Andric       // Byte size is enabled
1400b57cec5SDimitry Andric       if (byte_size == 0)
1410b57cec5SDimitry Andric         ParserGDBFormatLetter(execution_context, m_prev_gdb_size, format,
1420b57cec5SDimitry Andric                               byte_size);
1430b57cec5SDimitry Andric     } else {
1440b57cec5SDimitry Andric       // Byte size is disabled, make sure it wasn't specified but if this is an
1450b57cec5SDimitry Andric       // address, it's actually necessary to specify one so don't error out
1460b57cec5SDimitry Andric       if (byte_size > 0 && format != lldb::eFormatAddressInfo) {
1470b57cec5SDimitry Andric         error.SetErrorString(
1480b57cec5SDimitry Andric             "this command doesn't support specifying a byte size");
1490b57cec5SDimitry Andric         return error;
1500b57cec5SDimitry Andric       }
1510b57cec5SDimitry Andric     }
1520b57cec5SDimitry Andric 
1530b57cec5SDimitry Andric     if (count_enabled) {
1540b57cec5SDimitry Andric       // Count is enabled and was not set, set it to the default for gdb format
1550b57cec5SDimitry Andric       // statements (which is 1).
1560b57cec5SDimitry Andric       if (count == 0)
1570b57cec5SDimitry Andric         count = 1;
1580b57cec5SDimitry Andric     } else {
1590b57cec5SDimitry Andric       // Count is disabled, make sure it wasn't specified
1600b57cec5SDimitry Andric       if (count > 0) {
1610b57cec5SDimitry Andric         error.SetErrorString("this command doesn't support specifying a count");
1620b57cec5SDimitry Andric         return error;
1630b57cec5SDimitry Andric       }
1640b57cec5SDimitry Andric     }
1650b57cec5SDimitry Andric 
1660b57cec5SDimitry Andric     m_format.SetCurrentValue(format);
1670b57cec5SDimitry Andric     m_format.SetOptionWasSet();
1680b57cec5SDimitry Andric     if (byte_size_enabled) {
1690b57cec5SDimitry Andric       m_byte_size.SetCurrentValue(byte_size);
1700b57cec5SDimitry Andric       m_byte_size.SetOptionWasSet();
1710b57cec5SDimitry Andric     }
1720b57cec5SDimitry Andric     if (count_enabled) {
1730b57cec5SDimitry Andric       m_count.SetCurrentValue(count);
1740b57cec5SDimitry Andric       m_count.SetOptionWasSet();
1750b57cec5SDimitry Andric     }
1760b57cec5SDimitry Andric   } break;
1770b57cec5SDimitry Andric 
1780b57cec5SDimitry Andric   default:
1799dba64beSDimitry Andric     llvm_unreachable("Unimplemented option");
1800b57cec5SDimitry Andric   }
1810b57cec5SDimitry Andric 
1820b57cec5SDimitry Andric   return error;
1830b57cec5SDimitry Andric }
1840b57cec5SDimitry Andric 
ParserGDBFormatLetter(ExecutionContext * execution_context,char format_letter,Format & format,uint32_t & byte_size)1850b57cec5SDimitry Andric bool OptionGroupFormat::ParserGDBFormatLetter(
1860b57cec5SDimitry Andric     ExecutionContext *execution_context, char format_letter, Format &format,
1870b57cec5SDimitry Andric     uint32_t &byte_size) {
1880b57cec5SDimitry Andric   m_has_gdb_format = true;
1890b57cec5SDimitry Andric   switch (format_letter) {
1900b57cec5SDimitry Andric   case 'o':
1910b57cec5SDimitry Andric     format = eFormatOctal;
1920b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
1930b57cec5SDimitry Andric     return true;
1940b57cec5SDimitry Andric   case 'x':
1950b57cec5SDimitry Andric     format = eFormatHex;
1960b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
1970b57cec5SDimitry Andric     return true;
1980b57cec5SDimitry Andric   case 'd':
1990b57cec5SDimitry Andric     format = eFormatDecimal;
2000b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2010b57cec5SDimitry Andric     return true;
2020b57cec5SDimitry Andric   case 'u':
2030b57cec5SDimitry Andric     format = eFormatUnsigned;
2040b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2050b57cec5SDimitry Andric     return true;
2060b57cec5SDimitry Andric   case 't':
2070b57cec5SDimitry Andric     format = eFormatBinary;
2080b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2090b57cec5SDimitry Andric     return true;
2100b57cec5SDimitry Andric   case 'f':
2110b57cec5SDimitry Andric     format = eFormatFloat;
2120b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2130b57cec5SDimitry Andric     return true;
2140b57cec5SDimitry Andric   case 'a':
2150b57cec5SDimitry Andric     format = eFormatAddressInfo;
2160b57cec5SDimitry Andric     {
2170b57cec5SDimitry Andric       TargetSP target_sp =
2180b57cec5SDimitry Andric           execution_context ? execution_context->GetTargetSP() : TargetSP();
2190b57cec5SDimitry Andric       if (target_sp)
2200b57cec5SDimitry Andric         byte_size = target_sp->GetArchitecture().GetAddressByteSize();
2210b57cec5SDimitry Andric       m_prev_gdb_format = format_letter;
2220b57cec5SDimitry Andric       return true;
2230b57cec5SDimitry Andric     }
2240b57cec5SDimitry Andric   case 'i':
2250b57cec5SDimitry Andric     format = eFormatInstruction;
2260b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2270b57cec5SDimitry Andric     return true;
2280b57cec5SDimitry Andric   case 'c':
2290b57cec5SDimitry Andric     format = eFormatChar;
2300b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2310b57cec5SDimitry Andric     return true;
2320b57cec5SDimitry Andric   case 's':
2330b57cec5SDimitry Andric     format = eFormatCString;
2340b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2350b57cec5SDimitry Andric     return true;
2360b57cec5SDimitry Andric   case 'T':
2370b57cec5SDimitry Andric     format = eFormatOSType;
2380b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2390b57cec5SDimitry Andric     return true;
2400b57cec5SDimitry Andric   case 'A':
2410b57cec5SDimitry Andric     format = eFormatHexFloat;
2420b57cec5SDimitry Andric     m_prev_gdb_format = format_letter;
2430b57cec5SDimitry Andric     return true;
2440b57cec5SDimitry Andric 
2450b57cec5SDimitry Andric   case 'b':
2460b57cec5SDimitry Andric   case 'h':
2470b57cec5SDimitry Andric   case 'w':
2480b57cec5SDimitry Andric   case 'g':
2490b57cec5SDimitry Andric     {
2500b57cec5SDimitry Andric       // Size isn't used for printing instructions, so if a size is specified,
2510b57cec5SDimitry Andric       // and the previous format was 'i', then we should reset it to the
2520b57cec5SDimitry Andric       // default ('x').  Otherwise we'll continue to print as instructions,
2530b57cec5SDimitry Andric       // which isn't expected.
2540b57cec5SDimitry Andric       if (format_letter == 'b')
2550b57cec5SDimitry Andric           byte_size = 1;
2560b57cec5SDimitry Andric       else if (format_letter == 'h')
2570b57cec5SDimitry Andric           byte_size = 2;
2580b57cec5SDimitry Andric       else if (format_letter == 'w')
2590b57cec5SDimitry Andric           byte_size = 4;
2600b57cec5SDimitry Andric       else if (format_letter == 'g')
2610b57cec5SDimitry Andric           byte_size = 8;
2620b57cec5SDimitry Andric 
2630b57cec5SDimitry Andric         m_prev_gdb_size = format_letter;
2640b57cec5SDimitry Andric         if (m_prev_gdb_format == 'i')
2650b57cec5SDimitry Andric           m_prev_gdb_format = 'x';
2660b57cec5SDimitry Andric         return true;
2670b57cec5SDimitry Andric     }
2680b57cec5SDimitry Andric     break;
2690b57cec5SDimitry Andric   default:
2700b57cec5SDimitry Andric     break;
2710b57cec5SDimitry Andric   }
2720b57cec5SDimitry Andric 
2730b57cec5SDimitry Andric 
2740b57cec5SDimitry Andric   return false;
2750b57cec5SDimitry Andric }
2760b57cec5SDimitry Andric 
OptionParsingStarting(ExecutionContext * execution_context)2770b57cec5SDimitry Andric void OptionGroupFormat::OptionParsingStarting(
2780b57cec5SDimitry Andric     ExecutionContext *execution_context) {
2790b57cec5SDimitry Andric   m_format.Clear();
2800b57cec5SDimitry Andric   m_byte_size.Clear();
2810b57cec5SDimitry Andric   m_count.Clear();
2820b57cec5SDimitry Andric   m_has_gdb_format = false;
2830b57cec5SDimitry Andric }
284