/* * Copyright (c) 2018 Stefan Sperling * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include "got_error.h" #include "got_object.h" #include "got_lib_delta.h" #include "got_lib_inflate.h" #include "got_lib_object.h" #include "got_lib_sha1.h" #ifndef nitems #define nitems(_a) (sizeof(_a) / sizeof((_a)[0])) #endif const struct got_error * got_error(int code) { int i; for (i = 0; i < nitems(got_errors); i++) { if (code == got_errors[i].code) return &got_errors[i]; } abort(); } const struct got_error * got_error_msg(int code, const char *msg) { static struct got_error err; int i; for (i = 0; i < nitems(got_errors); i++) { if (code == got_errors[i].code) { err.code = code; err.msg = msg; return &err; } } abort(); } const struct got_error * got_error_from_errno(const char *prefix) { static struct got_error err; static char err_msg[PATH_MAX + 20]; snprintf(err_msg, sizeof(err_msg), "%s: %s", prefix, strerror(errno)); err.code = GOT_ERR_ERRNO; err.msg = err_msg; return &err; } const struct got_error * got_error_from_errno2(const char *prefix, const char *prefix2) { static struct got_error err; static char err_msg[(PATH_MAX * 2) + 20]; snprintf(err_msg, sizeof(err_msg), "%s: %s: %s", prefix, prefix2, strerror(errno)); err.code = GOT_ERR_ERRNO; err.msg = err_msg; return &err; } const struct got_error * got_error_from_errno3(const char *prefix, const char *prefix2, const char *prefix3) { static struct got_error err; static char err_msg[(PATH_MAX * 3) + 20]; snprintf(err_msg, sizeof(err_msg), "%s: %s: %s: %s", prefix, prefix2, prefix3, strerror(errno)); err.code = GOT_ERR_ERRNO; err.msg = err_msg; return &err; } const struct got_error * got_error_set_errno(int code, const char *prefix) { errno = code; return got_error_from_errno(prefix); } const struct got_error * got_ferror(FILE *f, int code) { if (ferror(f)) return got_error_from_errno(""); return got_error(code); } const struct got_error * got_error_no_obj(struct got_object_id *id) { static char msg[sizeof("object not found") + SHA1_DIGEST_STRING_LENGTH]; char id_str[SHA1_DIGEST_STRING_LENGTH]; int ret; if (!got_sha1_digest_to_str(id->sha1, id_str, sizeof(id_str))) return got_error(GOT_ERR_NO_OBJ); ret = snprintf(msg, sizeof(msg), "object %s not found", id_str); if (ret == -1 || ret >= sizeof(msg)) return got_error(GOT_ERR_NO_OBJ); return got_error_msg(GOT_ERR_NO_OBJ, msg); } const struct got_error * got_error_not_ref(const char *refname) { static char msg[sizeof("reference not found") + 1004]; int ret; ret = snprintf(msg, sizeof(msg), "reference %s not found", refname); if (ret == -1 || ret >= sizeof(msg)) return got_error(GOT_ERR_NOT_REF); return got_error_msg(GOT_ERR_NOT_REF, msg); } const struct got_error * got_error_uuid(uint32_t uuid_status, const char *prefix) { switch (uuid_status) { case uuid_s_ok: return NULL; case uuid_s_bad_version: return got_error(GOT_ERR_UUID_VERSION); case uuid_s_invalid_string_uuid: return got_error(GOT_ERR_UUID_INVALID); case uuid_s_no_memory: return got_error_set_errno(ENOMEM, prefix); default: return got_error(GOT_ERR_UUID); } } const struct got_error * got_error_path(const char *path, int code) { static struct got_error err; static char msg[PATH_MAX + 128]; int i; for (i = 0; i < nitems(got_errors); i++) { if (code == got_errors[i].code) { err.code = code; snprintf(msg, sizeof(msg), "%s: %s", path, got_errors[i].msg); err.msg = msg; return &err; } } abort(); }