blob: 6104a3884a4b4387c7f2de09393a0b44c46d2047 [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
ludi6e8eccd2017-08-14 14:40:37 -070020#include "binder/Status.h"
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090021#include <cassert>
22#include <ostream>
23
24namespace android {
25namespace netdutils {
26
27// Simple status implementation suitable for use on the stack in low
28// or moderate performance code. This can definitely be improved but
29// for now short string optimization is expected to keep the common
30// success case fast.
31class Status {
32 public:
33 Status() = default;
34
Jonathan Basseri0e9d5fa2017-08-22 11:39:17 -070035 explicit Status(int code) : mCode(code) {}
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090036
37 Status(int code, const std::string& msg) : mCode(code), mMsg(msg) { assert(!ok()); }
38
39 int code() const { return mCode; }
40
41 bool ok() const { return code() == 0; }
42
43 const std::string& msg() const { return mMsg; }
44
Joel Scherpelzde937962017-06-01 13:20:21 +090045 bool operator==(const Status& other) const { return code() == other.code(); }
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090046 bool operator!=(const Status& other) const { return !(*this == other); }
47
48 private:
49 int mCode = 0;
50 std::string mMsg;
51};
52
53namespace status {
54
55const Status ok{0};
Joel Scherpelzde937962017-06-01 13:20:21 +090056// EOF is not part of errno space, we'll place it far above the
57// highest existing value.
58const Status eof{0x10001, "end of file"};
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090059const Status undefined{std::numeric_limits<int>::max(), "undefined"};
60
61} // namespace status
62
63// Return true if status is "OK". This is sometimes preferable to
64// status.ok() when we want to check the state of Status-like objects
65// that implicitly cast to Status.
Joel Scherpelzde937962017-06-01 13:20:21 +090066inline bool isOk(const Status& status) {
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090067 return status.ok();
68}
69
70// Document that status is expected to be ok. This function may log
71// (or assert when running in debug mode) if status has an unexpected
72// value.
Joel Scherpelzde937962017-06-01 13:20:21 +090073void expectOk(const Status& status);
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090074
75// Convert POSIX errno to a Status object.
76// If Status is extended to have more features, this mapping may
77// become more complex.
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090078Status statusFromErrno(int err, const std::string& msg);
79
Joel Scherpelzde937962017-06-01 13:20:21 +090080// Helper that checks Status-like object (notably StatusOr) against a
81// value in the errno space.
82bool equalToErrno(const Status& status, int err);
83
ludi6e8eccd2017-08-14 14:40:37 -070084// Converts netdutils Status into binder Status.
85binder::Status asBinderStatus(const netdutils::Status& status);
86
Joel Scherpelzde937962017-06-01 13:20:21 +090087// Helper that converts Status-like object (notably StatusOr) to a
88// message.
89std::string toString(const Status& status);
Joel Scherpelz08b84cd2017-05-22 13:11:54 +090090
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090091std::ostream& operator<<(std::ostream& os, const Status& s);
92
Jonathan Basseri774f0062017-08-22 10:40:27 -070093// Evaluate 'stmt' to a Status object and if it results in an error, return that
94// error. Use 'tmp' as a variable name to avoid shadowing any variables named
95// tmp.
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +090096#define RETURN_IF_NOT_OK_IMPL(tmp, stmt) \
97 do { \
98 ::android::netdutils::Status tmp = (stmt); \
99 if (!isOk(tmp)) { \
100 return tmp; \
101 } \
102 } while (false)
103
Jonathan Basseri774f0062017-08-22 10:40:27 -0700104// Create a unique variable name to avoid shadowing local variables.
Joel Scherpelzf3fa5cc2017-05-22 12:30:03 +0900105#define RETURN_IF_NOT_OK_CONCAT(line, stmt) RETURN_IF_NOT_OK_IMPL(__CONCAT(_status_, line), stmt)
106
107// Macro to allow exception-like handling of error return values.
108//
109// If the evaluation of stmt results in an error, return that error
110// from current function.
111//
112// Example usage:
113// Status bar() { ... }
114//
115// RETURN_IF_NOT_OK(status);
116// RETURN_IF_NOT_OK(bar());
117#define RETURN_IF_NOT_OK(stmt) RETURN_IF_NOT_OK_CONCAT(__LINE__, stmt)
118
119} // namespace netdutils
120} // namespace android
121
122#endif /* NETUTILS_STATUS_H */