xref: /aosp_15_r20/art/cmdline/cmdline_parse_result.h (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #ifndef ART_CMDLINE_CMDLINE_PARSE_RESULT_H_
18*795d594fSAndroid Build Coastguard Worker #define ART_CMDLINE_CMDLINE_PARSE_RESULT_H_
19*795d594fSAndroid Build Coastguard Worker 
20*795d594fSAndroid Build Coastguard Worker #include "cmdline_result.h"
21*795d594fSAndroid Build Coastguard Worker #include "detail/cmdline_parser_detail.h"
22*795d594fSAndroid Build Coastguard Worker 
23*795d594fSAndroid Build Coastguard Worker namespace art {
24*795d594fSAndroid Build Coastguard Worker // Result of a type-parsing attempt. If successful holds the strongly-typed value,
25*795d594fSAndroid Build Coastguard Worker // otherwise it holds either a usage or a failure string message that should be displayed back
26*795d594fSAndroid Build Coastguard Worker // to the user.
27*795d594fSAndroid Build Coastguard Worker //
28*795d594fSAndroid Build Coastguard Worker // CmdlineType::Parse/CmdlineType::ParseAndAppend must return this type.
29*795d594fSAndroid Build Coastguard Worker template <typename T>
30*795d594fSAndroid Build Coastguard Worker struct CmdlineParseResult : CmdlineResult {
31*795d594fSAndroid Build Coastguard Worker   using CmdlineResult::CmdlineResult;
32*795d594fSAndroid Build Coastguard Worker 
33*795d594fSAndroid Build Coastguard Worker   // Create an error result with the usage error code and the specified message.
UsageCmdlineParseResult34*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult Usage(const std::string& message) {
35*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(kUsage, message);
36*795d594fSAndroid Build Coastguard Worker   }
37*795d594fSAndroid Build Coastguard Worker 
38*795d594fSAndroid Build Coastguard Worker   // Create an error result with the failure error code and no message.
FailureCmdlineParseResult39*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> Failure()  {
40*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(kFailure);
41*795d594fSAndroid Build Coastguard Worker   }
42*795d594fSAndroid Build Coastguard Worker 
43*795d594fSAndroid Build Coastguard Worker   // Create an error result with the failure error code and no message.
FailureCmdlineParseResult44*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> Failure(const std::string& message) {
45*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(kFailure, message);
46*795d594fSAndroid Build Coastguard Worker   }
47*795d594fSAndroid Build Coastguard Worker 
48*795d594fSAndroid Build Coastguard Worker   // Create a successful result which holds the specified value.
SuccessCmdlineParseResult49*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> Success(const T& value) {
50*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(value);
51*795d594fSAndroid Build Coastguard Worker   }
52*795d594fSAndroid Build Coastguard Worker 
53*795d594fSAndroid Build Coastguard Worker   // Create a successful result, taking over the value.
SuccessCmdlineParseResult54*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> Success(T&& value) {
55*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(std::forward<T>(value));
56*795d594fSAndroid Build Coastguard Worker   }
57*795d594fSAndroid Build Coastguard Worker 
58*795d594fSAndroid Build Coastguard Worker   // Create succesful result, without any values. Used when a value was successfully appended
59*795d594fSAndroid Build Coastguard Worker   // into an existing object.
SuccessNoValueCmdlineParseResult60*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> SuccessNoValue() {
61*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(T {});
62*795d594fSAndroid Build Coastguard Worker   }
63*795d594fSAndroid Build Coastguard Worker 
64*795d594fSAndroid Build Coastguard Worker   // Create an error result with the OutOfRange error and the specified message.
OutOfRangeCmdlineParseResult65*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> OutOfRange(const std::string& message) {
66*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(kOutOfRange, message);
67*795d594fSAndroid Build Coastguard Worker   }
68*795d594fSAndroid Build Coastguard Worker 
69*795d594fSAndroid Build Coastguard Worker   // Create an error result with the OutOfRange code and a custom message
70*795d594fSAndroid Build Coastguard Worker   // which is printed from the actual/min/max values.
71*795d594fSAndroid Build Coastguard Worker   // Values are converted to string using the ostream<< operator.
OutOfRangeCmdlineParseResult72*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> OutOfRange(const T& value,
73*795d594fSAndroid Build Coastguard Worker                                           const T& min,
74*795d594fSAndroid Build Coastguard Worker                                           const T& max) {
75*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult(kOutOfRange,
76*795d594fSAndroid Build Coastguard Worker                               "actual: " + art::detail::ToStringAny(value) +
77*795d594fSAndroid Build Coastguard Worker                               ", min: " + art::detail::ToStringAny(min) +
78*795d594fSAndroid Build Coastguard Worker                               ", max: " + art::detail::ToStringAny(max));
79*795d594fSAndroid Build Coastguard Worker   }
80*795d594fSAndroid Build Coastguard Worker 
81*795d594fSAndroid Build Coastguard Worker   // Get a read-only reference to the underlying value.
82*795d594fSAndroid Build Coastguard Worker   // The result must have been successful and must have a value.
GetValueCmdlineParseResult83*795d594fSAndroid Build Coastguard Worker   const T& GetValue() const {
84*795d594fSAndroid Build Coastguard Worker     assert(IsSuccess());
85*795d594fSAndroid Build Coastguard Worker     assert(has_value_);
86*795d594fSAndroid Build Coastguard Worker     return value_;
87*795d594fSAndroid Build Coastguard Worker   }
88*795d594fSAndroid Build Coastguard Worker 
89*795d594fSAndroid Build Coastguard Worker   // Get a mutable reference to the underlying value.
90*795d594fSAndroid Build Coastguard Worker   // The result must have been successful and must have a value.
GetValueCmdlineParseResult91*795d594fSAndroid Build Coastguard Worker   T& GetValue() {
92*795d594fSAndroid Build Coastguard Worker     assert(IsSuccess());
93*795d594fSAndroid Build Coastguard Worker     assert(has_value_);
94*795d594fSAndroid Build Coastguard Worker     return value_;
95*795d594fSAndroid Build Coastguard Worker   }
96*795d594fSAndroid Build Coastguard Worker 
97*795d594fSAndroid Build Coastguard Worker   // Take over the value.
98*795d594fSAndroid Build Coastguard Worker   // The result must have been successful and must have a value.
ReleaseValueCmdlineParseResult99*795d594fSAndroid Build Coastguard Worker   T&& ReleaseValue() {
100*795d594fSAndroid Build Coastguard Worker     assert(IsSuccess());
101*795d594fSAndroid Build Coastguard Worker     assert(has_value_);
102*795d594fSAndroid Build Coastguard Worker     return std::move(value_);
103*795d594fSAndroid Build Coastguard Worker   }
104*795d594fSAndroid Build Coastguard Worker 
105*795d594fSAndroid Build Coastguard Worker   // Whether or not the result has a value (e.g. created with Result::Success).
106*795d594fSAndroid Build Coastguard Worker   // Error results never have values, success results commonly, but not always, have values.
HasValueCmdlineParseResult107*795d594fSAndroid Build Coastguard Worker   bool HasValue() const {
108*795d594fSAndroid Build Coastguard Worker     return has_value_;
109*795d594fSAndroid Build Coastguard Worker   }
110*795d594fSAndroid Build Coastguard Worker 
111*795d594fSAndroid Build Coastguard Worker   // Cast an error-result from type T2 to T1.
112*795d594fSAndroid Build Coastguard Worker   // Safe since error-results don't store a typed value.
113*795d594fSAndroid Build Coastguard Worker   template <typename T2>
CastErrorCmdlineParseResult114*795d594fSAndroid Build Coastguard Worker   static CmdlineParseResult<T> CastError(const CmdlineParseResult<T2>& other) {
115*795d594fSAndroid Build Coastguard Worker     assert(other.IsError());
116*795d594fSAndroid Build Coastguard Worker     return CmdlineParseResult<T>(other.GetStatus());
117*795d594fSAndroid Build Coastguard Worker   }
118*795d594fSAndroid Build Coastguard Worker 
119*795d594fSAndroid Build Coastguard Worker   // Make sure copying is allowed
120*795d594fSAndroid Build Coastguard Worker   CmdlineParseResult(const CmdlineParseResult&) = default;
121*795d594fSAndroid Build Coastguard Worker   // Make sure moving is cheap
122*795d594fSAndroid Build Coastguard Worker   CmdlineParseResult(CmdlineParseResult&&) noexcept = default;
123*795d594fSAndroid Build Coastguard Worker 
124*795d594fSAndroid Build Coastguard Worker  private:
CmdlineParseResultCmdlineParseResult125*795d594fSAndroid Build Coastguard Worker   explicit CmdlineParseResult(const T& value)
126*795d594fSAndroid Build Coastguard Worker     : CmdlineResult(kSuccess), value_(value), has_value_(true) {}
CmdlineParseResultCmdlineParseResult127*795d594fSAndroid Build Coastguard Worker   explicit CmdlineParseResult(T&& value)
128*795d594fSAndroid Build Coastguard Worker     : CmdlineResult(kSuccess), value_(std::forward<T>(value)), has_value_(true) {}
CmdlineParseResultCmdlineParseResult129*795d594fSAndroid Build Coastguard Worker   CmdlineParseResult()
130*795d594fSAndroid Build Coastguard Worker     : CmdlineResult(kSuccess), value_(), has_value_(false) {}
131*795d594fSAndroid Build Coastguard Worker 
132*795d594fSAndroid Build Coastguard Worker   T value_;
133*795d594fSAndroid Build Coastguard Worker   bool has_value_ = false;
134*795d594fSAndroid Build Coastguard Worker };
135*795d594fSAndroid Build Coastguard Worker 
136*795d594fSAndroid Build Coastguard Worker }  // namespace art
137*795d594fSAndroid Build Coastguard Worker 
138*795d594fSAndroid Build Coastguard Worker #endif  // ART_CMDLINE_CMDLINE_PARSE_RESULT_H_
139