1 // Copyright 2014 The Chromium OS Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include <brillo/errors/error_codes.h>
6
7 #include <base/posix/safe_strerror.h>
8
9 namespace brillo {
10 namespace errors {
11
12 namespace dbus {
13 const char kDomain[] = "dbus";
14 } // namespace dbus
15
16 namespace json {
17 const char kDomain[] = "json_parser";
18 const char kParseError[] = "json_parse_error";
19 const char kObjectExpected[] = "json_object_expected";
20 } // namespace json
21
22 namespace http {
23 const char kDomain[] = "http";
24 } // namespace http
25
26 namespace system {
27 const char kDomain[] = "system";
28
29 namespace {
30 const struct ErrorMapEntry {
31 const char* error_code;
32 int errnum;
33 } error_map[] = {
34 #define ERROR_ENTRY(err) { #err, err }
35 ERROR_ENTRY(EPERM), // Operation not permitted
36 ERROR_ENTRY(ENOENT), // No such file or directory
37 ERROR_ENTRY(ESRCH), // No such process
38 ERROR_ENTRY(EINTR), // Interrupted system call
39 ERROR_ENTRY(EIO), // I/O error
40 ERROR_ENTRY(ENXIO), // No such device or address
41 ERROR_ENTRY(E2BIG), // Argument list too long
42 ERROR_ENTRY(ENOEXEC), // Exec format error
43 ERROR_ENTRY(EBADF), // Bad file number
44 ERROR_ENTRY(ECHILD), // No child processes
45 ERROR_ENTRY(EAGAIN), // Try again
46 ERROR_ENTRY(ENOMEM), // Out of memory
47 ERROR_ENTRY(EACCES), // Permission denied
48 ERROR_ENTRY(EFAULT), // Bad address
49 ERROR_ENTRY(ENOTBLK), // Block device required
50 ERROR_ENTRY(EBUSY), // Device or resource busy
51 ERROR_ENTRY(EEXIST), // File exists
52 ERROR_ENTRY(EXDEV), // Cross-device link
53 ERROR_ENTRY(ENODEV), // No such device
54 ERROR_ENTRY(ENOTDIR), // Not a directory
55 ERROR_ENTRY(EISDIR), // Is a directory
56 ERROR_ENTRY(EINVAL), // Invalid argument
57 ERROR_ENTRY(ENFILE), // File table overflow
58 ERROR_ENTRY(EMFILE), // Too many open files
59 ERROR_ENTRY(ENOTTY), // Not a typewriter
60 ERROR_ENTRY(ETXTBSY), // Text file busy
61 ERROR_ENTRY(EFBIG), // File too large
62 ERROR_ENTRY(ENOSPC), // No space left on device
63 ERROR_ENTRY(ESPIPE), // Illegal seek
64 ERROR_ENTRY(EROFS), // Read-only file system
65 ERROR_ENTRY(EMLINK), // Too many links
66 ERROR_ENTRY(EPIPE), // Broken pipe
67 ERROR_ENTRY(EDOM), // Math argument out of domain of func
68 ERROR_ENTRY(ERANGE), // Math result not representable
69 ERROR_ENTRY(EDEADLK), // Resource deadlock would occur
70 ERROR_ENTRY(ENAMETOOLONG), // File name too long
71 ERROR_ENTRY(ENOLCK), // No record locks available
72 ERROR_ENTRY(ENOSYS), // Function not implemented
73 ERROR_ENTRY(ENOTEMPTY), // Directory not empty
74 ERROR_ENTRY(ELOOP), // Too many symbolic links encountered
75 ERROR_ENTRY(ENOMSG), // No message of desired type
76 ERROR_ENTRY(EIDRM), // Identifier removed
77 #ifdef __linux__
78 ERROR_ENTRY(ECHRNG), // Channel number out of range
79 ERROR_ENTRY(EL2NSYNC), // Level 2 not synchronized
80 ERROR_ENTRY(EL3HLT), // Level 3 halted
81 ERROR_ENTRY(EL3RST), // Level 3 reset
82 ERROR_ENTRY(ELNRNG), // Link number out of range
83 ERROR_ENTRY(EUNATCH), // Protocol driver not attached
84 ERROR_ENTRY(ENOCSI), // No CSI structure available
85 ERROR_ENTRY(EL2HLT), // Level 2 halted
86 ERROR_ENTRY(EBADE), // Invalid exchange
87 ERROR_ENTRY(EBADR), // Invalid request descriptor
88 ERROR_ENTRY(EXFULL), // Exchange full
89 ERROR_ENTRY(ENOANO), // No anode
90 ERROR_ENTRY(EBADRQC), // Invalid request code
91 ERROR_ENTRY(EBADSLT), // Invalid slot
92 ERROR_ENTRY(EBFONT), // Bad font file format
93 #endif // __linux__
94 ERROR_ENTRY(ENOSTR), // Device not a stream
95 ERROR_ENTRY(ENODATA), // No data available
96 ERROR_ENTRY(ETIME), // Timer expired
97 ERROR_ENTRY(ENOSR), // Out of streams resources
98 #ifdef __linux__
99 ERROR_ENTRY(ENONET), // Machine is not on the network
100 ERROR_ENTRY(ENOPKG), // Package not installed
101 #endif // __linux__
102 ERROR_ENTRY(EREMOTE), // Object is remote
103 ERROR_ENTRY(ENOLINK), // Link has been severed
104 #ifdef __linux__
105 ERROR_ENTRY(EADV), // Advertise error
106 ERROR_ENTRY(ESRMNT), // Srmount error
107 ERROR_ENTRY(ECOMM), // Communication error on send
108 #endif // __linux__
109 ERROR_ENTRY(EPROTO), // Protocol error
110 ERROR_ENTRY(EMULTIHOP), // Multihop attempted
111 #ifdef __linux__
112 ERROR_ENTRY(EDOTDOT), // RFS specific error
113 #endif // __linux__
114 ERROR_ENTRY(EBADMSG), // Not a data message
115 ERROR_ENTRY(EOVERFLOW), // Value too large for defined data type
116 #ifdef __linux__
117 ERROR_ENTRY(ENOTUNIQ), // Name not unique on network
118 ERROR_ENTRY(EBADFD), // File descriptor in bad state
119 ERROR_ENTRY(EREMCHG), // Remote address changed
120 ERROR_ENTRY(ELIBACC), // Can not access a needed shared library
121 ERROR_ENTRY(ELIBBAD), // Accessing a corrupted shared library
122 ERROR_ENTRY(ELIBSCN), // .lib section in a.out corrupted
123 ERROR_ENTRY(ELIBMAX), // Attempting to link in too many shared libs.
124 ERROR_ENTRY(ELIBEXEC), // Cannot exec a shared library directly
125 #endif // __linux__
126 ERROR_ENTRY(EILSEQ), // Illegal byte sequence
127 #ifdef __linux__
128 ERROR_ENTRY(ERESTART), // Interrupted system call should be restarted
129 ERROR_ENTRY(ESTRPIPE), // Streams pipe error
130 #endif // __linux__
131 ERROR_ENTRY(EUSERS), // Too many users
132 ERROR_ENTRY(ENOTSOCK), // Socket operation on non-socket
133 ERROR_ENTRY(EDESTADDRREQ), // Destination address required
134 ERROR_ENTRY(EMSGSIZE), // Message too long
135 ERROR_ENTRY(EPROTOTYPE), // Protocol wrong type for socket
136 ERROR_ENTRY(ENOPROTOOPT), // Protocol not available
137 ERROR_ENTRY(EPROTONOSUPPORT), // Protocol not supported
138 ERROR_ENTRY(ESOCKTNOSUPPORT), // Socket type not supported
139 ERROR_ENTRY(EOPNOTSUPP), // Operation not supported o/transport endpoint
140 ERROR_ENTRY(EPFNOSUPPORT), // Protocol family not supported
141 ERROR_ENTRY(EAFNOSUPPORT), // Address family not supported by protocol
142 ERROR_ENTRY(EADDRINUSE), // Address already in use
143 ERROR_ENTRY(EADDRNOTAVAIL), // Cannot assign requested address
144 ERROR_ENTRY(ENETDOWN), // Network is down
145 ERROR_ENTRY(ENETUNREACH), // Network is unreachable
146 ERROR_ENTRY(ENETRESET), // Network dropped connection because of reset
147 ERROR_ENTRY(ECONNABORTED), // Software caused connection abort
148 ERROR_ENTRY(ECONNRESET), // Connection reset by peer
149 ERROR_ENTRY(ENOBUFS), // No buffer space available
150 ERROR_ENTRY(EISCONN), // Transport endpoint is already connected
151 ERROR_ENTRY(ENOTCONN), // Transport endpoint is not connected
152 ERROR_ENTRY(ESHUTDOWN), // Cannot send after transp. endpoint shutdown
153 ERROR_ENTRY(ETOOMANYREFS), // Too many references: cannot splice
154 ERROR_ENTRY(ETIMEDOUT), // Connection timed out
155 ERROR_ENTRY(ECONNREFUSED), // Connection refused
156 ERROR_ENTRY(EHOSTDOWN), // Host is down
157 ERROR_ENTRY(EHOSTUNREACH), // No route to host
158 ERROR_ENTRY(EALREADY), // Operation already in progress
159 ERROR_ENTRY(EINPROGRESS), // Operation now in progress
160 ERROR_ENTRY(ESTALE), // Stale file handle
161 #ifdef __linux__
162 ERROR_ENTRY(EUCLEAN), // Structure needs cleaning
163 ERROR_ENTRY(ENOTNAM), // Not a XENIX named type file
164 ERROR_ENTRY(ENAVAIL), // No XENIX semaphores available
165 ERROR_ENTRY(EISNAM), // Is a named type file
166 ERROR_ENTRY(EREMOTEIO), // Remote I/O error
167 #endif // __linux__
168 ERROR_ENTRY(EDQUOT), // Quota exceeded
169 #ifdef __linux__
170 ERROR_ENTRY(ENOMEDIUM), // No medium found
171 ERROR_ENTRY(EMEDIUMTYPE), // Wrong medium type
172 #endif // __linux__
173 ERROR_ENTRY(ECANCELED), // Operation Canceled
174 #ifdef __linux__
175 ERROR_ENTRY(ENOKEY), // Required key not available
176 ERROR_ENTRY(EKEYEXPIRED), // Key has expired
177 ERROR_ENTRY(EKEYREVOKED), // Key has been revoked
178 ERROR_ENTRY(EKEYREJECTED), // Key was rejected by service
179 #endif // __linux__
180 ERROR_ENTRY(EOWNERDEAD), // Owner died
181 ERROR_ENTRY(ENOTRECOVERABLE), // State not recoverable
182 #ifdef __linux__
183 ERROR_ENTRY(ERFKILL), // Operation not possible due to RF-kill
184 ERROR_ENTRY(EHWPOISON), // Memory page has hardware error
185 #endif // __linux__
186 #undef ERROR_ENTRY
187 // This list comes from <errno.h> system header. The elements are ordered
188 // by increasing errnum values which is the same order used in the header
189 // file. So, when new error codes are added to glibc, it should be relatively
190 // easy to identify them and add them to this list.
191 };
192
193 // Gets the error code string from system error code. If unknown system error
194 // number is provided, returns an empty string.
ErrorCodeFromSystemError(int errnum)195 std::string ErrorCodeFromSystemError(int errnum) {
196 std::string error_code;
197 for (const ErrorMapEntry& entry : error_map) {
198 if (entry.errnum == errnum) {
199 error_code = entry.error_code;
200 break;
201 }
202 }
203 return error_code;
204 }
205
206 } // anonymous namespace
207
AddSystemError(ErrorPtr * error,const base::Location & location,int errnum)208 void AddSystemError(ErrorPtr* error,
209 const base::Location& location,
210 int errnum) {
211 std::string message = base::safe_strerror(errnum);
212 std::string code = ErrorCodeFromSystemError(errnum);
213 if (message.empty())
214 message = "Unknown error " + std::to_string(errnum);
215
216 if (code.empty())
217 code = "error_" + std::to_string(errnum);
218
219 Error::AddTo(error, location, kDomain, code, message);
220 }
221
222 } // namespace system
223
224 } // namespace errors
225 } // namespace brillo
226