1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2013 Lennart Poettering
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
31 #include "errno-list.h"
32 #include "string-util.h"
34 #include "bus-error.h"
36 BUS_ERROR_MAP_ELF_REGISTER
const sd_bus_error_map bus_standard_errors
[] = {
37 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.Failed", EACCES
),
38 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NoMemory", ENOMEM
),
39 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.ServiceUnknown", EHOSTUNREACH
),
40 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NameHasNoOwner", ENXIO
),
41 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NoReply", ETIMEDOUT
),
42 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.IOError", EIO
),
43 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.BadAddress", EADDRNOTAVAIL
),
44 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NotSupported", EOPNOTSUPP
),
45 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.LimitsExceeded", ENOBUFS
),
46 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.AccessDenied", EACCES
),
47 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.AuthFailed", EACCES
),
48 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.InteractiveAuthorizationRequired", EACCES
),
49 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NoServer", EHOSTDOWN
),
50 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.Timeout", ETIMEDOUT
),
51 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.NoNetwork", ENONET
),
52 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.AddressInUse", EADDRINUSE
),
53 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.Disconnected", ECONNRESET
),
54 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.InvalidArgs", EINVAL
),
55 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.FileNotFound", ENOENT
),
56 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.FileExists", EEXIST
),
57 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.UnknownMethod", EBADR
),
58 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.UnknownObject", EBADR
),
59 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.UnknownInterface", EBADR
),
60 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.UnknownProperty", EBADR
),
61 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.PropertyReadOnly", EROFS
),
62 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.UnixProcessIdUnknown", ESRCH
),
63 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.InvalidSignature", EINVAL
),
64 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.InconsistentMessage", EBADMSG
),
65 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.TimedOut", ETIMEDOUT
),
66 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.MatchRuleInvalid", EINVAL
),
67 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.InvalidFileContent", EINVAL
),
68 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.MatchRuleNotFound", ENOENT
),
69 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.SELinuxSecurityContextUnknown", ESRCH
),
70 SD_BUS_ERROR_MAP("org.freedesktop.DBus.Error.ObjectPathInUse", EBUSY
),
74 /* GCC maps this magically to the beginning and end of the BUS_ERROR_MAP section.
75 * Hide them; for currently unknown reasons they get exported to the shared libries
76 * even without being listed in the sym file. */
77 extern const sd_bus_error_map __start_BUS_ERROR_MAP
[] _hidden_
;
78 extern const sd_bus_error_map __stop_BUS_ERROR_MAP
[] _hidden_
;
80 /* Additional maps registered with sd_bus_error_add_map() are in this
81 * NULL terminated array */
82 static const sd_bus_error_map
**additional_error_maps
= NULL
;
84 static int bus_error_name_to_errno(const char *name
) {
85 const sd_bus_error_map
**map
, *m
;
92 p
= startswith(name
, "System.Error.");
94 r
= errno_from_name(p
);
101 if (additional_error_maps
) {
102 for (map
= additional_error_maps
; *map
; map
++) {
103 for (m
= *map
;; m
++) {
104 /* For additional error maps the end marker is actually the end marker */
105 if (m
->code
== BUS_ERROR_MAP_END_MARKER
)
108 if (streq(m
->name
, name
))
114 m
= __start_BUS_ERROR_MAP
;
115 while (m
< __stop_BUS_ERROR_MAP
) {
116 /* For magic ELF error maps, the end marker might
117 * appear in the middle of things, since multiple maps
118 * might appear in the same section. Hence, let's skip
119 * over it, but realign the pointer to the netx 8byte
120 * boundary, which is the selected alignment for the
122 if (m
->code
== BUS_ERROR_MAP_END_MARKER
) {
127 if (streq(m
->name
, name
))
136 static sd_bus_error
errno_to_bus_error_const(int error
) {
144 return BUS_ERROR_OOM
;
148 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_ACCESS_DENIED
, "Access denied");
151 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_INVALID_ARGS
, "Invalid argument");
154 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_UNIX_PROCESS_ID_UNKNOWN
, "No such process");
157 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_FILE_NOT_FOUND
, "File not found");
160 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_FILE_EXISTS
, "File exists");
164 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_TIMEOUT
, "Timed out");
167 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_IO_ERROR
, "Input/output error");
172 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_DISCONNECTED
, "Disconnected");
175 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_NOT_SUPPORTED
, "Not supported");
178 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_BAD_ADDRESS
, "Address not available");
181 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_LIMITS_EXCEEDED
, "Limits exceeded");
184 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_ADDRESS_IN_USE
, "Address in use");
187 return SD_BUS_ERROR_MAKE_CONST(SD_BUS_ERROR_INCONSISTENT_MESSAGE
, "Inconsistent message");
190 return SD_BUS_ERROR_NULL
;
193 static int errno_to_bus_error_name_new(int error
, char **ret
) {
200 name
= errno_to_name(error
);
204 n
= strappend("System.Error.", name
);
212 bool bus_error_is_dirty(sd_bus_error
*e
) {
216 return e
->name
|| e
->message
|| e
->_need_free
!= 0;
219 _public_
void sd_bus_error_free(sd_bus_error
*e
) {
223 if (e
->_need_free
> 0) {
224 free((void*) e
->name
);
225 free((void*) e
->message
);
228 e
->name
= e
->message
= NULL
;
232 _public_
int sd_bus_error_set(sd_bus_error
*e
, const char *name
, const char *message
) {
239 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
241 e
->name
= strdup(name
);
248 e
->message
= strdup(message
);
253 return -bus_error_name_to_errno(name
);
256 int bus_error_setfv(sd_bus_error
*e
, const char *name
, const char *format
, va_list ap
) {
263 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
265 e
->name
= strdup(name
);
271 /* If we hit OOM on formatting the pretty message, we ignore
272 * this, since we at least managed to write the error name */
274 (void) vasprintf((char**) &e
->message
, format
, ap
);
279 return -bus_error_name_to_errno(name
);
282 _public_
int sd_bus_error_setf(sd_bus_error
*e
, const char *name
, const char *format
, ...) {
288 va_start(ap
, format
);
289 r
= bus_error_setfv(e
, name
, format
, ap
);
295 return sd_bus_error_set(e
, name
, NULL
);
298 _public_
int sd_bus_error_copy(sd_bus_error
*dest
, const sd_bus_error
*e
) {
300 if (!sd_bus_error_is_set(e
))
305 assert_return(!bus_error_is_dirty(dest
), -EINVAL
);
308 * _need_free < 0 indicates that the error is temporarily const, needs deep copying
309 * _need_free == 0 indicates that the error is perpetually const, needs no deep copying
310 * _need_free > 0 indicates that the error is fully dynamic, needs deep copying
313 if (e
->_need_free
== 0)
316 dest
->name
= strdup(e
->name
);
318 *dest
= BUS_ERROR_OOM
;
323 dest
->message
= strdup(e
->message
);
325 dest
->_need_free
= 1;
329 return -bus_error_name_to_errno(e
->name
);
332 _public_
int sd_bus_error_set_const(sd_bus_error
*e
, const char *name
, const char *message
) {
338 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
340 *e
= SD_BUS_ERROR_MAKE_CONST(name
, message
);
343 return -bus_error_name_to_errno(name
);
346 _public_
int sd_bus_error_is_set(const sd_bus_error
*e
) {
353 _public_
int sd_bus_error_has_name(const sd_bus_error
*e
, const char *name
) {
357 return streq_ptr(e
->name
, name
);
360 _public_
int sd_bus_error_get_errno(const sd_bus_error
* e
) {
367 return bus_error_name_to_errno(e
->name
);
370 static void bus_error_strerror(sd_bus_error
*e
, int error
) {
384 x
= strerror_r(error
, m
, k
);
385 if (errno
== ERANGE
|| strlen(x
) >= k
- 1) {
397 if (e
->_need_free
> 0) {
398 /* Error is already dynamic, let's just update the message */
399 free((char*) e
->message
);
404 /* Error was const so far, let's make it dynamic, if we can */
419 if (e
->_need_free
> 0) {
422 /* Error is dynamic, let's hence make the message also dynamic */
427 free((char*) e
->message
);
430 /* Error is const, hence we can just override */
439 _public_
int sd_bus_error_set_errno(sd_bus_error
*e
, int error
) {
449 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
451 /* First, try a const translation */
452 *e
= errno_to_bus_error_const(error
);
454 if (!sd_bus_error_is_set(e
)) {
457 /* If that didn't work, try a dynamic one. */
459 k
= errno_to_bus_error_name_new(error
, (char**) &e
->name
);
466 *e
= BUS_ERROR_FAILED
;
469 /* Now, fill in the message from strerror() if we can */
470 bus_error_strerror(e
, error
);
474 _public_
int sd_bus_error_set_errnofv(sd_bus_error
*e
, int error
, const char *format
, va_list ap
) {
486 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
488 /* First, try a const translation */
489 *e
= errno_to_bus_error_const(error
);
491 if (!sd_bus_error_is_set(e
)) {
494 /* If that didn't work, try a dynamic one */
496 k
= errno_to_bus_error_name_new(error
, (char**) &e
->name
);
503 *e
= BUS_ERROR_FAILED
;
509 /* Then, let's try to fill in the supplied message */
511 errno
= error
; /* Make sure that %m resolves to the specified error */
512 r
= vasprintf(&m
, format
, ap
);
515 if (e
->_need_free
<= 0) {
528 free((char*) e
->message
);
535 /* If that didn't work, use strerror() for the message */
536 bus_error_strerror(e
, error
);
540 _public_
int sd_bus_error_set_errnof(sd_bus_error
*e
, int error
, const char *format
, ...) {
551 assert_return(!bus_error_is_dirty(e
), -EINVAL
);
556 va_start(ap
, format
);
557 r
= sd_bus_error_set_errnofv(e
, error
, format
, ap
);
563 return sd_bus_error_set_errno(e
, error
);
566 const char *bus_error_message(const sd_bus_error
*e
, int error
) {
569 /* Sometimes the D-Bus server is a little bit too verbose with
570 * its error messages, so let's override them here */
571 if (sd_bus_error_has_name(e
, SD_BUS_ERROR_ACCESS_DENIED
))
572 return "Access denied";
581 return strerror(error
);
584 _public_
int sd_bus_error_add_map(const sd_bus_error_map
*map
) {
585 const sd_bus_error_map
**maps
= NULL
;
588 assert_return(map
, -EINVAL
);
590 if (additional_error_maps
) {
592 if (additional_error_maps
[n
] == NULL
)
595 if (additional_error_maps
[n
] == map
)
600 maps
= realloc_multiply(additional_error_maps
, sizeof(struct sd_bus_error_map
*), n
+ 2);
608 additional_error_maps
= maps
;