Merge branch 'maint'
authorJunio C Hamano <gitster@pobox.com>
Tue, 16 Dec 2008 07:06:13 +0000 (23:06 -0800)
committerJunio C Hamano <gitster@pobox.com>
Tue, 16 Dec 2008 07:06:13 +0000 (23:06 -0800)
* maint:
fast-import: close pack before unlinking it
pager: do not dup2 stderr if it is already redirected
git-show: do not segfault when showing a bad tag

1  2 
builtin-log.c
fast-import.c
pager.c
diff --combined builtin-log.c
index 840daf907897391fb193706982e2274c2a9e1014,db71e0da745ee480f8705c81f2899daf1dc6c4f5..99d1137b081d2e14d75e3d4d1766c3bf921f1493
@@@ -14,6 -14,7 +14,6 @@@
  #include "tag.h"
  #include "reflog-walk.h"
  #include "patch-ids.h"
 -#include "refs.h"
  #include "run-command.h"
  #include "shortlog.h"
  
@@@ -24,10 -25,36 +24,10 @@@ static int default_show_root = 1
  static const char *fmt_patch_subject_prefix = "PATCH";
  static const char *fmt_pretty;
  
 -static void add_name_decoration(const char *prefix, const char *name, struct object *obj)
 -{
 -      int plen = strlen(prefix);
 -      int nlen = strlen(name);
 -      struct name_decoration *res = xmalloc(sizeof(struct name_decoration) + plen + nlen);
 -      memcpy(res->name, prefix, plen);
 -      memcpy(res->name + plen, name, nlen + 1);
 -      res->next = add_decoration(&name_decoration, obj, res);
 -}
 -
 -static int add_ref_decoration(const char *refname, const unsigned char *sha1, int flags, void *cb_data)
 -{
 -      struct object *obj = parse_object(sha1);
 -      if (!obj)
 -              return 0;
 -      add_name_decoration("", refname, obj);
 -      while (obj->type == OBJ_TAG) {
 -              obj = ((struct tag *)obj)->tagged;
 -              if (!obj)
 -                      break;
 -              add_name_decoration("tag: ", refname, obj);
 -      }
 -      return 0;
 -}
 -
  static void cmd_log_init(int argc, const char **argv, const char *prefix,
                      struct rev_info *rev)
  {
        int i;
 -      int decorate = 0;
  
        rev->abbrev = DEFAULT_ABBREV;
        rev->commit_format = CMIT_FMT_DEFAULT;
@@@ -37,7 -64,6 +37,7 @@@
        DIFF_OPT_SET(&rev->diffopt, RECURSIVE);
        rev->show_root_diff = default_show_root;
        rev->subject_prefix = fmt_patch_subject_prefix;
 +      DIFF_OPT_SET(&rev->diffopt, ALLOW_TEXTCONV);
  
        if (default_date_mode)
                rev->date_mode = parse_date_format(default_date_mode);
        for (i = 1; i < argc; i++) {
                const char *arg = argv[i];
                if (!strcmp(arg, "--decorate")) {
 -                      if (!decorate)
 -                              for_each_ref(add_ref_decoration, NULL);
 -                      decorate = 1;
 +                      load_ref_decorations();
 +                      rev->show_decorations = 1;
 +              } else if (!strcmp(arg, "--source")) {
 +                      rev->show_source = 1;
                } else
                        die("unrecognized argument: %s", arg);
        }
@@@ -192,11 -217,6 +192,11 @@@ static int cmd_log_walk(struct rev_inf
        if (rev->early_output)
                finish_early_output(rev);
  
 +      /*
 +       * For --check and --exit-code, the exit code is based on CHECK_FAILED
 +       * and HAS_CHANGES being accumulated in rev->diffopt, so be careful to
 +       * retain that state information if replacing rev->diffopt in this loop
 +       */
        while ((commit = get_revision(rev)) != NULL) {
                log_tree_commit(rev, commit);
                if (!rev->reflog_info) {
                free_commit_list(commit->parents);
                commit->parents = NULL;
        }
 -      return 0;
 +      if (rev->diffopt.output_format & DIFF_FORMAT_CHECKDIFF &&
 +          DIFF_OPT_TST(&rev->diffopt, CHECK_FAILED)) {
 +              return 02;
 +      }
 +      return diff_result_code(&rev->diffopt, 0);
  }
  
  static int git_log_config(const char *var, const char *value, void *cb)
@@@ -340,7 -356,13 +340,13 @@@ int cmd_show(int argc, const char **arg
                                        t->tag,
                                        diff_get_color_opt(&rev.diffopt, DIFF_RESET));
                        ret = show_object(o->sha1, 1, &rev);
-                       objects[i].item = parse_object(t->tagged->sha1);
+                       if (ret)
+                               break;
+                       o = parse_object(t->tagged->sha1);
+                       if (!o)
+                               ret = error("Could not read object %s",
+                                           sha1_to_hex(t->tagged->sha1));
+                       objects[i].item = o;
                        i--;
                        break;
                }
@@@ -428,7 -450,7 +434,7 @@@ static int istitlechar(char c
  
  static const char *fmt_patch_suffix = ".patch";
  static int numbered = 0;
 -static int auto_number = 0;
 +static int auto_number = 1;
  
  static char **extra_hdr;
  static int extra_hdr_nr;
@@@ -487,7 -509,6 +493,7 @@@ static int git_format_config(const cha
                        return 0;
                }
                numbered = git_config_bool(var, value);
 +              auto_number = auto_number && numbered;
                return 0;
        }
  
@@@ -631,9 -652,10 +637,9 @@@ static void gen_message_id(struct rev_i
        const char *committer = git_committer_info(IDENT_WARN_ON_NO_NAME);
        const char *email_start = strrchr(committer, '<');
        const char *email_end = strrchr(committer, '>');
 -      struct strbuf buf;
 +      struct strbuf buf = STRBUF_INIT;
        if (!email_start || !email_end || email_start > email_end - 1)
                die("Could not extract email from committer identity.");
 -      strbuf_init(&buf, 0);
        strbuf_addf(&buf, "%s.%lu.git.%.*s", base,
                    (unsigned long) time(NULL),
                    (int)(email_end - email_start - 1), email_start + 1);
@@@ -652,7 -674,7 +658,7 @@@ static void make_cover_letter(struct re
        const char *msg;
        const char *extra_headers = rev->extra_headers;
        struct shortlog log;
 -      struct strbuf sb;
 +      struct strbuf sb = STRBUF_INIT;
        int i;
        const char *encoding = "utf-8";
        struct diff_options opts;
        committer = git_committer_info(0);
  
        msg = body;
 -      strbuf_init(&sb, 0);
        pp_user_info(NULL, CMIT_FMT_EMAIL, &sb, committer, DATE_RFC2822,
                     encoding);
        pp_title_line(CMIT_FMT_EMAIL, &msg, &sb, subject_start, extra_headers,
@@@ -754,7 -777,7 +760,7 @@@ int cmd_format_patch(int argc, const ch
        const char *in_reply_to = NULL;
        struct patch_ids ids;
        char *add_signoff = NULL;
 -      struct strbuf buf;
 +      struct strbuf buf = STRBUF_INIT;
  
        git_config(git_format_config, NULL);
        init_revisions(&rev, prefix);
        }
        argc = j;
  
 -      strbuf_init(&buf, 0);
 -
        for (i = 0; i < extra_hdr_nr; i++) {
                strbuf_addstr(&buf, extra_hdr[i]);
                strbuf_addch(&buf, '\n');
        if (argc > 1)
                die ("unrecognized argument: %s", argv[1]);
  
 -      if (!rev.diffopt.output_format)
 +      if (!rev.diffopt.output_format
 +              || rev.diffopt.output_format == DIFF_FORMAT_PATCH)
                rev.diffopt.output_format = DIFF_FORMAT_DIFFSTAT | DIFF_FORMAT_SUMMARY | DIFF_FORMAT_PATCH;
  
        if (!DIFF_OPT_TST(&rev.diffopt, TEXT) && !no_binary_diff)
@@@ -1138,7 -1162,8 +1144,7 @@@ int cmd_cherry(int argc, const char **a
                        sign = '-';
  
                if (verbose) {
 -                      struct strbuf buf;
 -                      strbuf_init(&buf, 0);
 +                      struct strbuf buf = STRBUF_INIT;
                        pretty_print_commit(CMIT_FMT_ONELINE, commit,
                                            &buf, 0, NULL, NULL, 0, 0);
                        printf("%c %s %s\n", sign,
diff --combined fast-import.c
index 3276d5d7aa49aaf41cf9f9e3562a5bbd14b63c18,13b36147a46390aa9fe3461eead08be7dadc35cd..201d4ffa13696f857c3a7cf84ba41fae21c04a7b
@@@ -376,7 -376,7 +376,7 @@@ static void dump_marks_helper(FILE *, u
  
  static void write_crash_report(const char *err)
  {
 -      char *loc = git_path("fast_import_crash_%d", getpid());
 +      char *loc = git_path("fast_import_crash_%"PRIuMAX, (uintmax_t) getpid());
        FILE *rpt = fopen(loc, "w");
        struct branch *b;
        unsigned long lu;
        fprintf(stderr, "fast-import: dumping crash report to %s\n", loc);
  
        fprintf(rpt, "fast-import crash report:\n");
 -      fprintf(rpt, "    fast-import process: %d\n", getpid());
 -      fprintf(rpt, "    parent process     : %d\n", getppid());
 +      fprintf(rpt, "    fast-import process: %"PRIuMAX"\n", (uintmax_t) getpid());
 +      fprintf(rpt, "    parent process     : %"PRIuMAX"\n", (uintmax_t) getppid());
        fprintf(rpt, "    at %s\n", show_date(time(NULL), 0, DATE_LOCAL));
        fputc('\n', rpt);
  
@@@ -554,10 -554,6 +554,10 @@@ static void *pool_alloc(size_t len
        struct mem_pool *p;
        void *r;
  
 +      /* round up to a 'uintmax_t' alignment */
 +      if (len & (sizeof(uintmax_t) - 1))
 +              len += sizeof(uintmax_t) - (len & (sizeof(uintmax_t) - 1));
 +
        for (p = mem_pool; p; p = p->next_pool)
                if ((p->end - p->next_free >= len))
                        break;
        }
  
        r = p->next_free;
 -      /* round out to a 'uintmax_t' alignment */
 -      if (len & (sizeof(uintmax_t) - 1))
 -              len += sizeof(uintmax_t) - (len & (sizeof(uintmax_t) - 1));
        p->next_free += len;
        return r;
  }
@@@ -846,7 -845,7 +846,7 @@@ static int oecmp (const void *a_, cons
  static char *create_index(void)
  {
        static char tmpfile[PATH_MAX];
 -      SHA_CTX ctx;
 +      git_SHA_CTX ctx;
        struct sha1file *f;
        struct object_entry **idx, **c, **last, *e;
        struct object_entry_pool *o;
        idx_fd = xmkstemp(tmpfile);
        f = sha1fd(idx_fd, tmpfile);
        sha1write(f, array, 256 * sizeof(int));
 -      SHA1_Init(&ctx);
 +      git_SHA1_Init(&ctx);
        for (c = idx; c != last; c++) {
                uint32_t offset = htonl((*c)->offset);
                sha1write(f, &offset, 4);
                sha1write(f, (*c)->sha1, sizeof((*c)->sha1));
 -              SHA1_Update(&ctx, (*c)->sha1, 20);
 +              git_SHA1_Update(&ctx, (*c)->sha1, 20);
        }
        sha1write(f, pack_data->sha1, sizeof(pack_data->sha1));
        sha1close(f, NULL, CSUM_FSYNC);
        free(idx);
 -      SHA1_Final(pack_data->sha1, &ctx);
 +      git_SHA1_Final(pack_data->sha1, &ctx);
        return tmpfile;
  }
  
@@@ -983,8 -982,10 +983,10 @@@ static void end_packfile(void
  
                pack_id++;
        }
-       else
+       else {
+               close(old_p->pack_fd);
                unlink(old_p->pack_name);
+       }
        free(old_p);
  
        /* We can't carry a delta across packfiles. */
@@@ -1034,15 -1035,15 +1036,15 @@@ static int store_object
        unsigned char hdr[96];
        unsigned char sha1[20];
        unsigned long hdrlen, deltalen;
 -      SHA_CTX c;
 +      git_SHA_CTX c;
        z_stream s;
  
        hdrlen = sprintf((char*)hdr,"%s %lu", typename(type),
                (unsigned long)dat->len) + 1;
 -      SHA1_Init(&c);
 -      SHA1_Update(&c, hdr, hdrlen);
 -      SHA1_Update(&c, dat->buf, dat->len);
 -      SHA1_Final(sha1, &c);
 +      git_SHA1_Init(&c);
 +      git_SHA1_Update(&c, hdr, hdrlen);
 +      git_SHA1_Update(&c, dat->buf, dat->len);
 +      git_SHA1_Final(sha1, &c);
        if (sha1out)
                hashcpy(sha1out, sha1);
  
diff --combined pager.c
index aa0966c9c55566382bf32c946c0a1846f004125a,0b7e55f476ac17609d88821e78bd89a511f4b3aa..f19ddbc87df04f117cd5e39189c8322fd5f29d68
+++ b/pager.c
@@@ -1,5 -1,4 +1,5 @@@
  #include "cache.h"
 +#include "run-command.h"
  
  /*
   * This is split up from the rest of git so that we can do
@@@ -9,7 -8,7 +9,7 @@@
  static int spawned_pager;
  
  #ifndef __MINGW32__
 -static void run_pager(const char *pager)
 +static void pager_preexec(void)
  {
        /*
         * Work around bug in "less" by not starting it until we
        FD_SET(0, &in);
        select(1, &in, NULL, &in, NULL);
  
 -      execlp(pager, pager, NULL);
 -      execl("/bin/sh", "sh", "-c", pager, NULL);
 +      setenv("LESS", "FRSX", 0);
  }
 -#else
 -#include "run-command.h"
 +#endif
  
  static const char *pager_argv[] = { "sh", "-c", NULL, NULL };
 -static struct child_process pager_process = {
 -      .argv = pager_argv,
 -      .in = -1
 -};
 +static struct child_process pager_process;
 +
  static void wait_for_pager(void)
  {
        fflush(stdout);
        close(2);
        finish_command(&pager_process);
  }
 -#endif
  
  void setup_pager(void)
  {
 -#ifndef __MINGW32__
 -      pid_t pid;
 -      int fd[2];
 -#endif
        const char *pager = getenv("GIT_PAGER");
  
        if (!isatty(1))
  
        spawned_pager = 1; /* means we are emitting to terminal */
  
 -#ifndef __MINGW32__
 -      if (pipe(fd) < 0)
 -              return;
 -      pid = fork();
 -      if (pid < 0) {
 -              close(fd[0]);
 -              close(fd[1]);
 -              return;
 -      }
 -
 -      /* return in the child */
 -      if (!pid) {
 -              dup2(fd[1], 1);
 -              dup2(fd[1], 2);
 -              close(fd[0]);
 -              close(fd[1]);
 -              return;
 -      }
 -
 -      /* The original process turns into the PAGER */
 -      dup2(fd[0], 0);
 -      close(fd[0]);
 -      close(fd[1]);
 -
 -      setenv("LESS", "FRSX", 0);
 -      run_pager(pager);
 -      die("unable to execute pager '%s'", pager);
 -      exit(255);
 -#else
        /* spawn the pager */
        pager_argv[2] = pager;
 +      pager_process.argv = pager_argv;
 +      pager_process.in = -1;
 +#ifndef __MINGW32__
 +      pager_process.preexec_cb = pager_preexec;
 +#endif
        if (start_command(&pager_process))
                return;
  
        /* original process continues, but writes to the pipe */
        dup2(pager_process.in, 1);
-       dup2(pager_process.in, 2);
+       if (isatty(2))
+               dup2(pager_process.in, 2);
        close(pager_process.in);
  
        /* this makes sure that the parent terminates after the pager */
        atexit(wait_for_pager);
 -#endif
  }
  
  int pager_in_use(void)