1 //===-- OptionGroupFormat.cpp ---------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "lldb/Interpreter/OptionGroupFormat.h"
11 #include "lldb/Host/OptionParser.h"
12 #include "lldb/Interpreter/CommandInterpreter.h"
13 #include "lldb/Target/ExecutionContext.h"
14 #include "lldb/Target/Target.h"
17 using namespace lldb_private
;
19 static constexpr OptionDefinition g_default_option_definitions
[] = {
20 {LLDB_OPT_SET_1
, false, "format", 'f', OptionParser::eRequiredArgument
,
21 nullptr, {}, 0, eArgTypeFormat
,
22 "Specify a format to be used for display."},
23 {LLDB_OPT_SET_2
, false, "gdb-format", 'G', OptionParser::eRequiredArgument
,
24 nullptr, {}, 0, eArgTypeGDBFormat
,
25 "Specify a format using a GDB format specifier string."},
26 {LLDB_OPT_SET_3
, false, "size", 's', OptionParser::eRequiredArgument
,
27 nullptr, {}, 0, eArgTypeByteSize
,
28 "The size in bytes to use when displaying with the selected format."},
29 {LLDB_OPT_SET_4
, false, "count", 'c', OptionParser::eRequiredArgument
,
30 nullptr, {}, 0, eArgTypeCount
,
31 "The number of total items to display."},
34 OptionGroupFormat::OptionGroupFormat(
35 lldb::Format default_format
, uint64_t default_byte_size
,
36 uint64_t default_count
, OptionGroupFormatUsageTextVector usage_text_vector
)
37 : m_format(default_format
, default_format
),
38 m_byte_size(default_byte_size
, default_byte_size
),
39 m_count(default_count
, default_count
), m_prev_gdb_format('x'),
40 m_prev_gdb_size('w'), m_has_gdb_format(false) {
41 // Copy the default option definitions.
42 std::copy(std::begin(g_default_option_definitions
),
43 std::end(g_default_option_definitions
),
44 std::begin(m_option_definitions
));
46 for (auto usage_text_tuple
: usage_text_vector
) {
47 switch (std::get
<0>(usage_text_tuple
)) {
49 m_option_definitions
[0].usage_text
= std::get
<1>(usage_text_tuple
);
51 case eArgTypeByteSize
:
52 m_option_definitions
[2].usage_text
= std::get
<1>(usage_text_tuple
);
55 llvm_unreachable("Unimplemented option");
60 llvm::ArrayRef
<OptionDefinition
> OptionGroupFormat::GetDefinitions() {
61 auto result
= llvm::ArrayRef(m_option_definitions
);
62 if (m_byte_size
.GetDefaultValue() < UINT64_MAX
) {
63 if (m_count
.GetDefaultValue() < UINT64_MAX
)
66 return result
.take_front(3);
68 return result
.take_front(2);
71 Status
OptionGroupFormat::SetOptionValue(uint32_t option_idx
,
72 llvm::StringRef option_arg
,
73 ExecutionContext
*execution_context
) {
75 const int short_option
= m_option_definitions
[option_idx
].short_option
;
77 switch (short_option
) {
79 error
= m_format
.SetValueFromString(option_arg
);
83 if (m_count
.GetDefaultValue() == 0) {
84 error
.SetErrorString("--count option is disabled");
86 error
= m_count
.SetValueFromString(option_arg
);
87 if (m_count
.GetCurrentValue() == 0)
88 error
.SetErrorStringWithFormat("invalid --count option value '%s'",
89 option_arg
.str().c_str());
94 if (m_byte_size
.GetDefaultValue() == 0) {
95 error
.SetErrorString("--size option is disabled");
97 error
= m_byte_size
.SetValueFromString(option_arg
);
98 if (m_byte_size
.GetCurrentValue() == 0)
99 error
.SetErrorStringWithFormat("invalid --size option value '%s'",
100 option_arg
.str().c_str());
106 llvm::StringRef gdb_format_str
= option_arg
;
107 gdb_format_str
.consumeInteger(0, count
);
109 Format format
= eFormatDefault
;
110 uint32_t byte_size
= 0;
112 while (!gdb_format_str
.empty() &&
113 ParserGDBFormatLetter(execution_context
, gdb_format_str
[0], format
,
115 gdb_format_str
= gdb_format_str
.drop_front();
118 // We the first character of the "gdb_format_str" is not the
119 // NULL terminator, we didn't consume the entire string and
120 // something is wrong. Also, if none of the format, size or count was
121 // specified correctly, then abort.
122 if (!gdb_format_str
.empty() ||
123 (format
== eFormatInvalid
&& byte_size
== 0 && count
== 0)) {
124 // Nothing got set correctly
125 error
.SetErrorStringWithFormat("invalid gdb format string '%s'",
126 option_arg
.str().c_str());
130 // At least one of the format, size or count was set correctly. Anything
131 // that wasn't set correctly should be set to the previous default
132 if (format
== eFormatInvalid
)
133 ParserGDBFormatLetter(execution_context
, m_prev_gdb_format
, format
,
136 const bool byte_size_enabled
= m_byte_size
.GetDefaultValue() < UINT64_MAX
;
137 const bool count_enabled
= m_count
.GetDefaultValue() < UINT64_MAX
;
138 if (byte_size_enabled
) {
139 // Byte size is enabled
141 ParserGDBFormatLetter(execution_context
, m_prev_gdb_size
, format
,
144 // Byte size is disabled, make sure it wasn't specified but if this is an
145 // address, it's actually necessary to specify one so don't error out
146 if (byte_size
> 0 && format
!= lldb::eFormatAddressInfo
) {
147 error
.SetErrorString(
148 "this command doesn't support specifying a byte size");
154 // Count is enabled and was not set, set it to the default for gdb format
155 // statements (which is 1).
159 // Count is disabled, make sure it wasn't specified
161 error
.SetErrorString("this command doesn't support specifying a count");
166 m_format
.SetCurrentValue(format
);
167 m_format
.SetOptionWasSet();
168 if (byte_size_enabled
) {
169 m_byte_size
.SetCurrentValue(byte_size
);
170 m_byte_size
.SetOptionWasSet();
173 m_count
.SetCurrentValue(count
);
174 m_count
.SetOptionWasSet();
179 llvm_unreachable("Unimplemented option");
185 bool OptionGroupFormat::ParserGDBFormatLetter(
186 ExecutionContext
*execution_context
, char format_letter
, Format
&format
,
187 uint32_t &byte_size
) {
188 m_has_gdb_format
= true;
189 switch (format_letter
) {
191 format
= eFormatOctal
;
192 m_prev_gdb_format
= format_letter
;
196 m_prev_gdb_format
= format_letter
;
199 format
= eFormatDecimal
;
200 m_prev_gdb_format
= format_letter
;
203 format
= eFormatUnsigned
;
204 m_prev_gdb_format
= format_letter
;
207 format
= eFormatBinary
;
208 m_prev_gdb_format
= format_letter
;
211 format
= eFormatFloat
;
212 m_prev_gdb_format
= format_letter
;
215 format
= eFormatAddressInfo
;
218 execution_context
? execution_context
->GetTargetSP() : TargetSP();
220 byte_size
= target_sp
->GetArchitecture().GetAddressByteSize();
221 m_prev_gdb_format
= format_letter
;
225 format
= eFormatInstruction
;
226 m_prev_gdb_format
= format_letter
;
229 format
= eFormatChar
;
230 m_prev_gdb_format
= format_letter
;
233 format
= eFormatCString
;
234 m_prev_gdb_format
= format_letter
;
237 format
= eFormatOSType
;
238 m_prev_gdb_format
= format_letter
;
241 format
= eFormatHexFloat
;
242 m_prev_gdb_format
= format_letter
;
250 // Size isn't used for printing instructions, so if a size is specified,
251 // and the previous format was 'i', then we should reset it to the
252 // default ('x'). Otherwise we'll continue to print as instructions,
253 // which isn't expected.
254 if (format_letter
== 'b')
256 else if (format_letter
== 'h')
258 else if (format_letter
== 'w')
260 else if (format_letter
== 'g')
263 m_prev_gdb_size
= format_letter
;
264 if (m_prev_gdb_format
== 'i')
265 m_prev_gdb_format
= 'x';
277 void OptionGroupFormat::OptionParsingStarting(
278 ExecutionContext
*execution_context
) {
282 m_has_gdb_format
= false;