e4fa6d2f0315c6f5ae63c97d83bfab3a14e279b2
   1/*
   2 * GIT - The information manager from hell
   3 *
   4 * Copyright (C) Linus Torvalds, 2005
   5 */
   6#include "git-compat-util.h"
   7#include "cache.h"
   8
   9static FILE *error_handle;
  10
  11void vreportf(const char *prefix, const char *err, va_list params)
  12{
  13        char msg[4096];
  14        FILE *fh = error_handle ? error_handle : stderr;
  15        vsnprintf(msg, sizeof(msg), err, params);
  16        fprintf(fh, "%s%s\n", prefix, msg);
  17}
  18
  19static NORETURN void usage_builtin(const char *err, va_list params)
  20{
  21        vreportf("usage: ", err, params);
  22        exit(129);
  23}
  24
  25static NORETURN void die_builtin(const char *err, va_list params)
  26{
  27        vreportf("fatal: ", err, params);
  28        exit(128);
  29}
  30
  31static void error_builtin(const char *err, va_list params)
  32{
  33        vreportf("error: ", err, params);
  34}
  35
  36static void warn_builtin(const char *warn, va_list params)
  37{
  38        vreportf("warning: ", warn, params);
  39}
  40
  41static int die_is_recursing_builtin(void)
  42{
  43        static int dying;
  44        return dying++;
  45}
  46
  47/* If we are in a dlopen()ed .so write to a global variable would segfault
  48 * (ugh), so keep things static. */
  49static NORETURN_PTR void (*usage_routine)(const char *err, va_list params) = usage_builtin;
  50static NORETURN_PTR void (*die_routine)(const char *err, va_list params) = die_builtin;
  51static void (*error_routine)(const char *err, va_list params) = error_builtin;
  52static void (*warn_routine)(const char *err, va_list params) = warn_builtin;
  53static int (*die_is_recursing)(void) = die_is_recursing_builtin;
  54
  55void set_die_routine(NORETURN_PTR void (*routine)(const char *err, va_list params))
  56{
  57        die_routine = routine;
  58}
  59
  60void set_error_routine(void (*routine)(const char *err, va_list params))
  61{
  62        error_routine = routine;
  63}
  64
  65void set_die_is_recursing_routine(int (*routine)(void))
  66{
  67        die_is_recursing = routine;
  68}
  69
  70void set_error_handle(FILE *fh)
  71{
  72        error_handle = fh;
  73}
  74
  75void NORETURN usagef(const char *err, ...)
  76{
  77        va_list params;
  78
  79        va_start(params, err);
  80        usage_routine(err, params);
  81        va_end(params);
  82}
  83
  84void NORETURN usage(const char *err)
  85{
  86        usagef("%s", err);
  87}
  88
  89void NORETURN die(const char *err, ...)
  90{
  91        va_list params;
  92
  93        if (die_is_recursing()) {
  94                fputs("fatal: recursion detected in die handler\n", stderr);
  95                exit(128);
  96        }
  97
  98        va_start(params, err);
  99        die_routine(err, params);
 100        va_end(params);
 101}
 102
 103void NORETURN die_errno(const char *fmt, ...)
 104{
 105        va_list params;
 106        char fmt_with_err[1024];
 107        char str_error[256], *err;
 108        int i, j;
 109
 110        if (die_is_recursing()) {
 111                fputs("fatal: recursion detected in die_errno handler\n",
 112                        stderr);
 113                exit(128);
 114        }
 115
 116        err = strerror(errno);
 117        for (i = j = 0; err[i] && j < sizeof(str_error) - 1; ) {
 118                if ((str_error[j++] = err[i++]) != '%')
 119                        continue;
 120                if (j < sizeof(str_error) - 1) {
 121                        str_error[j++] = '%';
 122                } else {
 123                        /* No room to double the '%', so we overwrite it with
 124                         * '\0' below */
 125                        j--;
 126                        break;
 127                }
 128        }
 129        str_error[j] = 0;
 130        snprintf(fmt_with_err, sizeof(fmt_with_err), "%s: %s", fmt, str_error);
 131
 132        va_start(params, fmt);
 133        die_routine(fmt_with_err, params);
 134        va_end(params);
 135}
 136
 137#undef error
 138int error(const char *err, ...)
 139{
 140        va_list params;
 141
 142        va_start(params, err);
 143        error_routine(err, params);
 144        va_end(params);
 145        return -1;
 146}
 147
 148void warning(const char *warn, ...)
 149{
 150        va_list params;
 151
 152        va_start(params, warn);
 153        warn_routine(warn, params);
 154        va_end(params);
 155}