blob: 5a8928923d3d0beea70d70df0be6d0a79a8f9456 [file] [log] [blame]
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +09001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef NETUTILS_STATUS_H
18#define NETUTILS_STATUS_H
19
20#include <cassert>
21#include <ostream>
22
23namespace android {
24namespace netdutils {
25
26// Simple status implementation suitable for use on the stack in low
27// or moderate performance code. This can definitely be improved but
28// for now short string optimization is expected to keep the common
29// success case fast.
30class Status {
31 public:
32 Status() = default;
33
34 Status(int code) : mCode(code) {}
35
36 Status(int code, const std::string& msg) : mCode(code), mMsg(msg) { assert(!ok()); }
37
38 int code() const { return mCode; }
39
40 bool ok() const { return code() == 0; }
41
42 const std::string& msg() const { return mMsg; }
43
44 bool operator==(const Status& other) const {
45 return (code() == other.code()) && (msg() == other.msg());
46 }
47 bool operator!=(const Status& other) const { return !(*this == other); }
48
49 private:
50 int mCode = 0;
51 std::string mMsg;
52};
53
54namespace status {
55
56const Status ok{0};
57const Status eof{256, "end of file"};
58const Status undefined{std::numeric_limits<int>::max(), "undefined"};
59
60} // namespace status
61
62// Return true if status is "OK". This is sometimes preferable to
63// status.ok() when we want to check the state of Status-like objects
64// that implicitly cast to Status.
65inline bool isOk(const Status status) {
66 return status.ok();
67}
68
69// Document that status is expected to be ok. This function may log
70// (or assert when running in debug mode) if status has an unexpected
71// value.
72void expectOk(const Status status);
73
74// Convert POSIX errno to a Status object.
75// If Status is extended to have more features, this mapping may
76// become more complex.
77//
78// TODO: msg is only a placeholder for now
79Status statusFromErrno(int err, const std::string& msg);
80
Joel Scherpelz08b84cd2017-05-22 13:11:54 +090081std::string toString(const Status status);
82
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090083std::ostream& operator<<(std::ostream& os, const Status& s);
84
85#define RETURN_IF_NOT_OK_IMPL(tmp, stmt) \
86 do { \
87 ::android::netdutils::Status tmp = (stmt); \
88 if (!isOk(tmp)) { \
89 return tmp; \
90 } \
91 } while (false)
92
93#define RETURN_IF_NOT_OK_CONCAT(line, stmt) RETURN_IF_NOT_OK_IMPL(__CONCAT(_status_, line), stmt)
94
95// Macro to allow exception-like handling of error return values.
96//
97// If the evaluation of stmt results in an error, return that error
98// from current function.
99//
100// Example usage:
101// Status bar() { ... }
102//
103// RETURN_IF_NOT_OK(status);
104// RETURN_IF_NOT_OK(bar());
105#define RETURN_IF_NOT_OK(stmt) RETURN_IF_NOT_OK_CONCAT(__LINE__, stmt)
106
107} // namespace netdutils
108} // namespace android
109
110#endif /* NETUTILS_STATUS_H */