merge-cache.con commit [PATCH] Teach diff-tree about commit objects (c1fdf2a)
   1#include <sys/types.h>
   2#include <sys/wait.h>
   3
   4#include "cache.h"
   5
   6static const char *pgm = NULL;
   7static const char *arguments[5];
   8
   9static void run_program(void)
  10{
  11        int pid = fork(), status;
  12
  13        if (pid < 0)
  14                die("unable to fork");
  15        if (!pid) {
  16                execlp(pgm, arguments[0],
  17                            arguments[1],
  18                            arguments[2],
  19                            arguments[3],
  20                            arguments[4],
  21                            NULL);
  22                die("unable to execute '%s'", pgm);
  23        }
  24        if (waitpid(pid, &status, 0) < 0 || !WIFEXITED(status) || WEXITSTATUS(status))
  25                die("merge program failed");
  26}
  27
  28static int merge_entry(int pos, const char *path)
  29{
  30        int found;
  31        
  32        if (pos >= active_nr)
  33                die("merge-cache: %s not in the cache", path);
  34        arguments[0] = pgm;
  35        arguments[1] = "";
  36        arguments[2] = "";
  37        arguments[3] = "";
  38        arguments[4] = path;
  39        found = 0;
  40        do {
  41                static char hexbuf[4][60];
  42                struct cache_entry *ce = active_cache[pos];
  43                int stage = ce_stage(ce);
  44
  45                if (strcmp(ce->name, path))
  46                        break;
  47                found++;
  48                strcpy(hexbuf[stage], sha1_to_hex(ce->sha1));
  49                arguments[stage] = hexbuf[stage];
  50        } while (++pos < active_nr);
  51        if (!found)
  52                die("merge-cache: %s not in the cache", path);
  53        run_program();
  54        return found;
  55}
  56
  57static void merge_file(const char *path)
  58{
  59        int pos = cache_name_pos(path, strlen(path));
  60
  61        /*
  62         * If it already exists in the cache as stage0, it's
  63         * already merged and there is nothing to do.
  64         */
  65        if (pos < 0)
  66                merge_entry(-pos-1, path);
  67}
  68
  69static void merge_all(void)
  70{
  71        int i;
  72        for (i = 0; i < active_nr; i++) {
  73                struct cache_entry *ce = active_cache[i];
  74                if (!ce_stage(ce))
  75                        continue;
  76                i += merge_entry(i, ce->name)-1;
  77        }
  78}
  79
  80int main(int argc, char **argv)
  81{
  82        int i, force_file = 0;
  83
  84        if (argc < 3)
  85                usage("merge-cache <merge-program> (-a | <filename>*)");
  86
  87        read_cache();
  88
  89        pgm = argv[1];
  90        for (i = 2; i < argc; i++) {
  91                char *arg = argv[i];
  92                if (!force_file && *arg == '-') {
  93                        if (!strcmp(arg, "--")) {
  94                                force_file = 1;
  95                                continue;
  96                        }
  97                        if (!strcmp(arg, "-a")) {
  98                                merge_all();
  99                                continue;
 100                        }
 101                        die("merge-cache: unknown option %s", arg);
 102                }
 103                merge_file(arg);
 104        }
 105        return 0;
 106}