perl / Git.xson commit Git.pm: Handle failed commands' output (8b9150e)
   1/* By carefully stacking #includes here (even if WE don't really need them)
   2 * we strive to make the thing actually compile. Git header files aren't very
   3 * nice. Perl headers are one of the signs of the coming apocalypse. */
   4#include <ctype.h>
   5/* Ok, it hasn't been so bad so far. */
   6
   7/* libgit interface */
   8#include "../cache.h"
   9#include "../exec_cmd.h"
  10
  11#define die perlyshadow_die__
  12
  13/* XS and Perl interface */
  14#include "EXTERN.h"
  15#include "perl.h"
  16#include "XSUB.h"
  17
  18#include "ppport.h"
  19
  20#undef die
  21
  22
  23static char *
  24report_xs(const char *prefix, const char *err, va_list params)
  25{
  26        static char buf[4096];
  27        strcpy(buf, prefix);
  28        vsnprintf(buf + strlen(prefix), 4096 - strlen(prefix), err, params);
  29        return buf;
  30}
  31
  32void
  33die_xs(const char *err, va_list params)
  34{
  35        char *str;
  36        str = report_xs("fatal: ", err, params);
  37        croak(str);
  38}
  39
  40int
  41error_xs(const char *err, va_list params)
  42{
  43        char *str;
  44        str = report_xs("error: ", err, params);
  45        warn(str);
  46        return -1;
  47}
  48
  49
  50MODULE = Git            PACKAGE = Git
  51
  52PROTOTYPES: DISABLE
  53
  54
  55BOOT:
  56{
  57        set_error_routine(error_xs);
  58        set_die_routine(die_xs);
  59}
  60
  61
  62# /* TODO: xs_call_gate(). See Git.pm. */
  63
  64
  65const char *
  66xs_version()
  67CODE:
  68{
  69        RETVAL = GIT_VERSION;
  70}
  71OUTPUT:
  72        RETVAL
  73
  74
  75const char *
  76xs_exec_path()
  77CODE:
  78{
  79        RETVAL = git_exec_path();
  80}
  81OUTPUT:
  82        RETVAL
  83
  84
  85void
  86xs__execv_git_cmd(...)
  87CODE:
  88{
  89        const char **argv;
  90        int i;
  91
  92        argv = malloc(sizeof(const char *) * (items + 1));
  93        if (!argv)
  94                croak("malloc failed");
  95        for (i = 0; i < items; i++)
  96                argv[i] = strdup(SvPV_nolen(ST(i)));
  97        argv[i] = NULL;
  98
  99        execv_git_cmd(argv);
 100
 101        for (i = 0; i < items; i++)
 102                if (argv[i])
 103                        free((char *) argv[i]);
 104        free((char **) argv);
 105}
 106
 107char *
 108xs_hash_object(file, type = "blob")
 109        SV *file;
 110        char *type;
 111CODE:
 112{
 113        unsigned char sha1[20];
 114
 115        if (SvTYPE(file) == SVt_RV)
 116                file = SvRV(file);
 117
 118        if (SvTYPE(file) == SVt_PVGV) {
 119                /* Filehandle */
 120                PerlIO *pio;
 121
 122                pio = IoIFP(sv_2io(file));
 123                if (!pio)
 124                        croak("You passed me something weird - a dir glob?");
 125                /* XXX: I just hope PerlIO didn't read anything from it yet.
 126                 * --pasky */
 127                if (index_pipe(sha1, PerlIO_fileno(pio), type, 0))
 128                        croak("Unable to hash given filehandle");
 129                /* Avoid any nasty surprises. */
 130                PerlIO_close(pio);
 131
 132        } else {
 133                /* String */
 134                char *path = SvPV_nolen(file);
 135                int fd = open(path, O_RDONLY);
 136                struct stat st;
 137
 138                if (fd < 0 ||
 139                    fstat(fd, &st) < 0 ||
 140                    index_fd(sha1, fd, &st, 0, type))
 141                        croak("Unable to hash %s", path);
 142                close(fd);
 143        }
 144        RETVAL = sha1_to_hex(sha1);
 145}
 146OUTPUT:
 147        RETVAL