Jonathan Tan <jonathanta...@google.com> writes:

> +/*
> + * Write oid to the given struct child_process's stdin, starting it first if
> + * necessary.
> + */
> +static int write_oid(const struct object_id *oid, struct packed_git *pack,
> +                  uint32_t pos, void *data)
> +{
> +     struct child_process *cmd = data;
> +
> +     if (cmd->in == -1) {
> +             if (start_command(cmd))
> +                     die("Could not start pack-objects to repack promisor 
> objects");
> +     }
> +
> +     xwrite(cmd->in, oid_to_hex(oid), GIT_SHA1_HEXSZ);
> +     xwrite(cmd->in, "\n", 1);
> +     return 0;
> +}
> +
> +static void repack_promisor_objects(const struct pack_objects_args *args,
> +                                 struct string_list *names)
> +{
> +     struct child_process cmd = CHILD_PROCESS_INIT;
> +     FILE *out;
> +     struct strbuf line = STRBUF_INIT;
> +
> +     prepare_pack_objects(&cmd, args);
> +     cmd.in = -1;
> +
> +     /*
> +      * NEEDSWORK: Giving pack-objects only the OIDs without any ordering
> +      * hints may result in suboptimal deltas in the resulting pack. See if
> +      * the OIDs can be sent with fake paths such that pack-objects can use a
> +      * {type -> existing pack order} ordering when computing deltas instead
> +      * of a {type -> size} ordering, which may produce better deltas.
> +      */
> +     for_each_packed_object(write_oid, &cmd,
> +                            FOR_EACH_OBJECT_PROMISOR_ONLY);
> +
> +     if (cmd.in == -1)
> +             /* No packed objects; cmd was never started */
> +             return;
> +     close(cmd.in);
> +
> +     out = xfdopen(cmd.out, "r");

Hmm, it is clever to auto-start the pack-objects (and notice there
wasn't anything needing to pack).  Two things that are worth noting
are:

 - The code takes advantage of the fact that cmd.in being left as -1
   is a sign that start_command() was never called (well, it could
   be that it was called but failed to open pipe---but then we would
   have died inside write_oid(), so we can ignore taht case).  This
   is not relying on its implementation detail---cmd.in would be
   replaced by a real fd to the pipe if start_command() was called.

 - We know that pack-objects reads all its standard input through to
   the EOF before emitting a single byte to its standard output, and
   that is why we can use bidi pipe and not worry about deadlocking
   caused by not reading from cmd.out at all (before closing cmd.in,
   that is).

So I kind of like the cleverness of the design of this code.

> +     while (strbuf_getline_lf(&line, out) != EOF) {
> +             char *promisor_name;
> +             int fd;
> +             if (line.len != 40)
> +                     die("repack: Expecting 40 character sha1 lines only 
> from pack-objects.");
> +             string_list_append(names, line.buf);
> +
> +             /*
> +              * pack-objects creates the .pack and .idx files, but not the
> +              * .promisor file. Create the .promisor file, which is empty.
> +              */
> +             promisor_name = mkpathdup("%s-%s.promisor", packtmp,
> +                                       line.buf);
> +             fd = open(promisor_name, O_CREAT|O_EXCL|O_WRONLY, 0600);
> +             if (fd < 0)
> +                     die_errno("unable to create '%s'", promisor_name);
> +             close(fd);
> +             free(promisor_name);

For now, as we do not know what is the appropriate thing to leave in
the file, empty is fine, but perhaps in the future we may want to
carry forward the info from the existing .promisor file?  What does
it initially contain?  Transport invoked with from_promisor bit
seems to call index-pack with "--promisor" option with no argument,
so this is probably in line with that.  Do we in the future need to
teach "--promisor" option to pack-objects we invoke here, which will
be passed to the index-pack it internally performs, so that we do
not have to do this by hand here?

In any case, don't we need to update the doc for the "--promisor"
option of "index-pack"?  The same comment for the new options added
in the same topic, e.g. "--from-promisor" and "--no-dependents"
options added to "fetch-pack".  It seems that the topic that ended
at 0c16cd499d was done rather hastily and under-documented?

> @@ -440,6 +513,8 @@ int cmd_repack(int argc, const char **argv, const char 
> *prefix)
>  
>       /* End of pack replacement. */
>  
> +     reprepare_packed_git(the_repository);
> +

I do not think this call hurts, but why does this become suddenly
necessary with _this_ patch?  Is it an unrelated bugfix?

>       if (delete_redundant) {
>               int opts = 0;
>               string_list_sort(&names);

Thanks.

Reply via email to