xref: /aosp_15_r20/external/abseil-cpp/absl/status/internal/status_internal.h (revision 9356374a3709195abf420251b3e825997ff56c0f)
1*9356374aSAndroid Build Coastguard Worker // Copyright 2019 The Abseil Authors.
2*9356374aSAndroid Build Coastguard Worker //
3*9356374aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*9356374aSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*9356374aSAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*9356374aSAndroid Build Coastguard Worker //
7*9356374aSAndroid Build Coastguard Worker //      https://www.apache.org/licenses/LICENSE-2.0
8*9356374aSAndroid Build Coastguard Worker //
9*9356374aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*9356374aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*9356374aSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*9356374aSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*9356374aSAndroid Build Coastguard Worker // limitations under the License.
14*9356374aSAndroid Build Coastguard Worker #ifndef ABSL_STATUS_INTERNAL_STATUS_INTERNAL_H_
15*9356374aSAndroid Build Coastguard Worker #define ABSL_STATUS_INTERNAL_STATUS_INTERNAL_H_
16*9356374aSAndroid Build Coastguard Worker 
17*9356374aSAndroid Build Coastguard Worker #include <atomic>
18*9356374aSAndroid Build Coastguard Worker #include <cstdint>
19*9356374aSAndroid Build Coastguard Worker #include <memory>
20*9356374aSAndroid Build Coastguard Worker #include <string>
21*9356374aSAndroid Build Coastguard Worker #include <utility>
22*9356374aSAndroid Build Coastguard Worker 
23*9356374aSAndroid Build Coastguard Worker #include "absl/base/attributes.h"
24*9356374aSAndroid Build Coastguard Worker #include "absl/base/config.h"
25*9356374aSAndroid Build Coastguard Worker #include "absl/base/nullability.h"
26*9356374aSAndroid Build Coastguard Worker #include "absl/container/inlined_vector.h"
27*9356374aSAndroid Build Coastguard Worker #include "absl/strings/cord.h"
28*9356374aSAndroid Build Coastguard Worker #include "absl/strings/string_view.h"
29*9356374aSAndroid Build Coastguard Worker #include "absl/types/optional.h"
30*9356374aSAndroid Build Coastguard Worker 
31*9356374aSAndroid Build Coastguard Worker #ifndef SWIG
32*9356374aSAndroid Build Coastguard Worker // Disabled for SWIG as it doesn't parse attributes correctly.
33*9356374aSAndroid Build Coastguard Worker namespace absl {
34*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_BEGIN
35*9356374aSAndroid Build Coastguard Worker // Returned Status objects may not be ignored. Codesearch doesn't handle ifdefs
36*9356374aSAndroid Build Coastguard Worker // as part of a class definitions (b/6995610), so we use a forward declaration.
37*9356374aSAndroid Build Coastguard Worker //
38*9356374aSAndroid Build Coastguard Worker // TODO(b/176172494): ABSL_MUST_USE_RESULT should expand to the more strict
39*9356374aSAndroid Build Coastguard Worker // [[nodiscard]]. For now, just use [[nodiscard]] directly when it is available.
40*9356374aSAndroid Build Coastguard Worker #if ABSL_HAVE_CPP_ATTRIBUTE(nodiscard)
41*9356374aSAndroid Build Coastguard Worker class [[nodiscard]] ABSL_ATTRIBUTE_TRIVIAL_ABI Status;
42*9356374aSAndroid Build Coastguard Worker #else
43*9356374aSAndroid Build Coastguard Worker class ABSL_MUST_USE_RESULT ABSL_ATTRIBUTE_TRIVIAL_ABI Status;
44*9356374aSAndroid Build Coastguard Worker #endif
45*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_END
46*9356374aSAndroid Build Coastguard Worker }  // namespace absl
47*9356374aSAndroid Build Coastguard Worker #endif  // !SWIG
48*9356374aSAndroid Build Coastguard Worker 
49*9356374aSAndroid Build Coastguard Worker namespace absl {
50*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_BEGIN
51*9356374aSAndroid Build Coastguard Worker 
52*9356374aSAndroid Build Coastguard Worker enum class StatusCode : int;
53*9356374aSAndroid Build Coastguard Worker enum class StatusToStringMode : int;
54*9356374aSAndroid Build Coastguard Worker 
55*9356374aSAndroid Build Coastguard Worker namespace status_internal {
56*9356374aSAndroid Build Coastguard Worker 
57*9356374aSAndroid Build Coastguard Worker // Container for status payloads.
58*9356374aSAndroid Build Coastguard Worker struct Payload {
59*9356374aSAndroid Build Coastguard Worker   std::string type_url;
60*9356374aSAndroid Build Coastguard Worker   absl::Cord payload;
61*9356374aSAndroid Build Coastguard Worker };
62*9356374aSAndroid Build Coastguard Worker 
63*9356374aSAndroid Build Coastguard Worker using Payloads = absl::InlinedVector<Payload, 1>;
64*9356374aSAndroid Build Coastguard Worker 
65*9356374aSAndroid Build Coastguard Worker // Reference-counted representation of Status data.
66*9356374aSAndroid Build Coastguard Worker class StatusRep {
67*9356374aSAndroid Build Coastguard Worker  public:
StatusRep(absl::StatusCode code_arg,absl::string_view message_arg,std::unique_ptr<status_internal::Payloads> payloads_arg)68*9356374aSAndroid Build Coastguard Worker   StatusRep(absl::StatusCode code_arg, absl::string_view message_arg,
69*9356374aSAndroid Build Coastguard Worker             std::unique_ptr<status_internal::Payloads> payloads_arg)
70*9356374aSAndroid Build Coastguard Worker       : ref_(int32_t{1}),
71*9356374aSAndroid Build Coastguard Worker         code_(code_arg),
72*9356374aSAndroid Build Coastguard Worker         message_(message_arg),
73*9356374aSAndroid Build Coastguard Worker         payloads_(std::move(payloads_arg)) {}
74*9356374aSAndroid Build Coastguard Worker 
code()75*9356374aSAndroid Build Coastguard Worker   absl::StatusCode code() const { return code_; }
message()76*9356374aSAndroid Build Coastguard Worker   const std::string& message() const { return message_; }
77*9356374aSAndroid Build Coastguard Worker 
78*9356374aSAndroid Build Coastguard Worker   // Ref and unref are const to allow access through a const pointer, and are
79*9356374aSAndroid Build Coastguard Worker   // used during copying operations.
Ref()80*9356374aSAndroid Build Coastguard Worker   void Ref() const { ref_.fetch_add(1, std::memory_order_relaxed); }
81*9356374aSAndroid Build Coastguard Worker   void Unref() const;
82*9356374aSAndroid Build Coastguard Worker 
83*9356374aSAndroid Build Coastguard Worker   // Payload methods correspond to the same methods in absl::Status.
84*9356374aSAndroid Build Coastguard Worker   absl::optional<absl::Cord> GetPayload(absl::string_view type_url) const;
85*9356374aSAndroid Build Coastguard Worker   void SetPayload(absl::string_view type_url, absl::Cord payload);
86*9356374aSAndroid Build Coastguard Worker   struct EraseResult {
87*9356374aSAndroid Build Coastguard Worker     bool erased;
88*9356374aSAndroid Build Coastguard Worker     uintptr_t new_rep;
89*9356374aSAndroid Build Coastguard Worker   };
90*9356374aSAndroid Build Coastguard Worker   EraseResult ErasePayload(absl::string_view type_url);
91*9356374aSAndroid Build Coastguard Worker   void ForEachPayload(
92*9356374aSAndroid Build Coastguard Worker       absl::FunctionRef<void(absl::string_view, const absl::Cord&)> visitor)
93*9356374aSAndroid Build Coastguard Worker       const;
94*9356374aSAndroid Build Coastguard Worker 
95*9356374aSAndroid Build Coastguard Worker   std::string ToString(StatusToStringMode mode) const;
96*9356374aSAndroid Build Coastguard Worker 
97*9356374aSAndroid Build Coastguard Worker   bool operator==(const StatusRep& other) const;
98*9356374aSAndroid Build Coastguard Worker   bool operator!=(const StatusRep& other) const { return !(*this == other); }
99*9356374aSAndroid Build Coastguard Worker 
100*9356374aSAndroid Build Coastguard Worker   // Returns an equivalent heap allocated StatusRep with refcount 1.
101*9356374aSAndroid Build Coastguard Worker   //
102*9356374aSAndroid Build Coastguard Worker   // `this` is not safe to be used after calling as it may have been deleted.
103*9356374aSAndroid Build Coastguard Worker   absl::Nonnull<StatusRep*> CloneAndUnref() const;
104*9356374aSAndroid Build Coastguard Worker 
105*9356374aSAndroid Build Coastguard Worker  private:
106*9356374aSAndroid Build Coastguard Worker   mutable std::atomic<int32_t> ref_;
107*9356374aSAndroid Build Coastguard Worker   absl::StatusCode code_;
108*9356374aSAndroid Build Coastguard Worker 
109*9356374aSAndroid Build Coastguard Worker   // As an internal implementation detail, we guarantee that if status.message()
110*9356374aSAndroid Build Coastguard Worker   // is non-empty, then the resulting string_view is null terminated.
111*9356374aSAndroid Build Coastguard Worker   // This is required to implement 'StatusMessageAsCStr(...)'
112*9356374aSAndroid Build Coastguard Worker   std::string message_;
113*9356374aSAndroid Build Coastguard Worker   std::unique_ptr<status_internal::Payloads> payloads_;
114*9356374aSAndroid Build Coastguard Worker };
115*9356374aSAndroid Build Coastguard Worker 
116*9356374aSAndroid Build Coastguard Worker absl::StatusCode MapToLocalCode(int value);
117*9356374aSAndroid Build Coastguard Worker 
118*9356374aSAndroid Build Coastguard Worker // Returns a pointer to a newly-allocated string with the given `prefix`,
119*9356374aSAndroid Build Coastguard Worker // suitable for output as an error message in assertion/`CHECK()` failures.
120*9356374aSAndroid Build Coastguard Worker //
121*9356374aSAndroid Build Coastguard Worker // This is an internal implementation detail for Abseil logging.
122*9356374aSAndroid Build Coastguard Worker ABSL_ATTRIBUTE_PURE_FUNCTION
123*9356374aSAndroid Build Coastguard Worker absl::Nonnull<std::string*> MakeCheckFailString(
124*9356374aSAndroid Build Coastguard Worker     absl::Nonnull<const absl::Status*> status,
125*9356374aSAndroid Build Coastguard Worker     absl::Nonnull<const char*> prefix);
126*9356374aSAndroid Build Coastguard Worker 
127*9356374aSAndroid Build Coastguard Worker }  // namespace status_internal
128*9356374aSAndroid Build Coastguard Worker 
129*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_END
130*9356374aSAndroid Build Coastguard Worker }  // namespace absl
131*9356374aSAndroid Build Coastguard Worker 
132*9356374aSAndroid Build Coastguard Worker #endif  // ABSL_STATUS_INTERNAL_STATUS_INTERNAL_H_
133