On 12/08/18 06:11, Christian Couder wrote:
> From: Jeff King <p...@peff.net>
> 
> Hosting providers that allow users to "fork" existing
> repos want those forks to share as much disk space as
> possible.
> 
> Alternates are an existing solution to keep all the
> objects from all the forks into a unique central repo,
> but this can have some drawbacks. Especially when
> packing the central repo, deltas will be created
> between objects from different forks.
> 
> This can make cloning or fetching a fork much slower
> and much more CPU intensive as Git might have to
> compute new deltas for many objects to avoid sending
> objects from a different fork.
> 
> Because the inefficiency primarily arises when an
> object is delitified against another object that does

s/delitified/deltified/ ?

> not exist in the same fork, we partition objects into
> sets that appear in the same fork, and define
> "delta islands". When finding delta base, we do not
> allow an object outside the same island to be
> considered as its base.
> 
> So "delta islands" is a way to store objects from
> different forks in the same repo and packfile without
> having deltas between objects from different forks.
> 
> This patch implements the delta islands mechanism in
> "delta-islands.{c,h}", but does not yet make use of it.
> 
> A few new fields are added in 'struct object_entry'
> in "pack-objects.h" though.
> 
> The documentation will follow in a patch that actually
> uses delta islands in "builtin/pack-objects.c".
> 
> Signed-off-by: Jeff King <p...@peff.net>
> Signed-off-by: Christian Couder <chrisc...@tuxfamily.org>
> ---
>  Makefile        |   1 +
>  delta-islands.c | 493 ++++++++++++++++++++++++++++++++++++++++++++++++
>  delta-islands.h |  11 ++
>  pack-objects.h  |   4 +
>  4 files changed, 509 insertions(+)
>  create mode 100644 delta-islands.c
>  create mode 100644 delta-islands.h
> 
> diff --git a/Makefile b/Makefile
> index bc4fc8eeab..e7994888e8 100644
> --- a/Makefile
> +++ b/Makefile
> @@ -841,6 +841,7 @@ LIB_OBJS += csum-file.o
>  LIB_OBJS += ctype.o
>  LIB_OBJS += date.o
>  LIB_OBJS += decorate.o
> +LIB_OBJS += delta-islands.o
>  LIB_OBJS += diffcore-break.o
>  LIB_OBJS += diffcore-delta.o
>  LIB_OBJS += diffcore-order.o
> diff --git a/delta-islands.c b/delta-islands.c
> new file mode 100644
> index 0000000000..c0b0da6837
> --- /dev/null
> +++ b/delta-islands.c
> @@ -0,0 +1,493 @@
> +#include "cache.h"
> +#include "attr.h"
> +#include "object.h"
> +#include "blob.h"
> +#include "commit.h"
> +#include "tag.h"
> +#include "tree.h"
> +#include "delta.h"
> +#include "pack.h"
> +#include "tree-walk.h"
> +#include "diff.h"
> +#include "revision.h"
> +#include "list-objects.h"
> +#include "progress.h"
> +#include "refs.h"
> +#include "khash.h"

I was wondering how many copies of the inline functions
introduced by this header we had, so:

  $ nm git | grep ' t ' | cut -d' ' -f3 | sort | uniq -c | sort -rn | grep kh_
        3 kh_resize_sha1
        3 kh_put_sha1
        3 kh_init_sha1
        3 kh_get_sha1
        1 kh_resize_str
        1 kh_resize_sha1_pos
        1 kh_put_str
        1 kh_put_sha1_pos
        1 kh_init_str
        1 kh_init_sha1_pos
        1 kh_get_str
        1 kh_get_sha1_pos
        1 kh_destroy_sha1
  $ 

Looking at the individual object files, we see:

  $ nm pack-bitmap-write.o | grep ' t ' | grep kh_
  00000000000001cc t kh_get_sha1
  00000000000001b7 t kh_init_sha1
  000000000000085e t kh_put_sha1
  0000000000000310 t kh_resize_sha1
  $ 

So, the two instances of the sha1 hash-map are never
destroyed (kh_destroy_sha1 is not present in the object
file).

  $ nm pack-bitmap.o | grep ' t ' | grep kh_
  00000000000002d9 t kh_destroy_sha1
  000000000000032b t kh_get_sha1
  0000000000000daa t kh_get_sha1_pos
  00000000000002c4 t kh_init_sha1
  0000000000000d95 t kh_init_sha1_pos
  00000000000009bd t kh_put_sha1
  0000000000001432 t kh_put_sha1_pos
  000000000000046f t kh_resize_sha1
  0000000000000eee t kh_resize_sha1_pos
  $ 

The sha1_pos hash-map is not destroyed here.

  $ nm delta-islands.o | grep ' t ' | grep kh_
  00000000000002be t kh_get_sha1
  0000000000000e52 t kh_get_str
  00000000000002a9 t kh_init_sha1
  0000000000000e3d t kh_init_str
  0000000000000950 t kh_put_sha1
  00000000000014e4 t kh_put_str
  0000000000000402 t kh_resize_sha1
  0000000000000f96 t kh_resize_str
  $ 

And neither the sha1 or str hash-maps are destroyed here.
(That is not necessarily a problem, of course! ;-) )
   
> +#include "pack-bitmap.h"
> +#include "pack-objects.h"
> +#include "delta-islands.h"
> +#include "sha1-array.h"
> +#include "config.h"
> +
> +KHASH_INIT(str, const char *, void *, 1, kh_str_hash_func, kh_str_hash_equal)
> +
> +static khash_sha1 *island_marks;
> +static unsigned island_counter;
> +static unsigned island_counter_core;
> +
> +static kh_str_t *remote_islands;
> +
> +struct remote_island {
> +     uint64_t hash;
> +     struct oid_array oids;
> +};
> +
> +struct island_bitmap {
> +     uint32_t refcount;
> +     uint32_t bits[];

Use FLEX_ARRAY here? We are slowly moving toward requiring
certain C99 features, but I can't remember a flex array
weather-balloon patch.

> +};
> +
> +static uint32_t island_bitmap_size;
> +
> +/*
> + * Allocate a new bitmap; if "old" is not NULL, the new bitmap will be a copy
> + * of "old". Otherwise, the new bitmap is empty.
> + */
> +static struct island_bitmap *island_bitmap_new(const struct island_bitmap 
> *old)
> +{
> +     size_t size = sizeof(struct island_bitmap) + (island_bitmap_size * 4);
> +     struct island_bitmap *b = xcalloc(1, size);
> +
> +     if (old)
> +             memcpy(b, old, size);
> +
> +     b->refcount = 1;
> +     return b;
> +}
> +
> +static void island_bitmap_or(struct island_bitmap *a, const struct 
> island_bitmap *b)
> +{
> +     uint32_t i;
> +
> +     for (i = 0; i < island_bitmap_size; ++i)
> +             a->bits[i] |= b->bits[i];
> +}
> +
> +static int island_bitmap_is_subset(struct island_bitmap *self,
> +             struct island_bitmap *super)
> +{
> +     uint32_t i;
> +
> +     if (self == super)
> +             return 1;
> +
> +     for (i = 0; i < island_bitmap_size; ++i) {
> +             if ((self->bits[i] & super->bits[i]) != self->bits[i])
> +                     return 0;
> +     }
> +
> +     return 1;
> +}
> +
> +#define ISLAND_BITMAP_BLOCK(x) (x / 32)
> +#define ISLAND_BITMAP_MASK(x) (1 << (x % 32))
> +
> +static void island_bitmap_set(struct island_bitmap *self, uint32_t i)
> +{
> +     self->bits[ISLAND_BITMAP_BLOCK(i)] |= ISLAND_BITMAP_MASK(i);
> +}
> +
> +static int island_bitmap_get(struct island_bitmap *self, uint32_t i)
> +{
> +     return (self->bits[ISLAND_BITMAP_BLOCK(i)] & ISLAND_BITMAP_MASK(i)) != 
> 0;
> +}
> +
> +int in_same_island(const struct object_id *trg_oid, const struct object_id 
> *src_oid)

Hmm, what does the trg_ prefix stand for?

> +{
> +     khiter_t trg_pos, src_pos;
> +
> +     /* If we aren't using islands, assume everything goes together. */
> +     if (!island_marks)
> +             return 1;
> +
> +     /*
> +      * If we don't have a bitmap for the target, we can delta it

... Ah, OK, trg_ => target.

> +      * against anything -- it's not an important object
> +      */
> +     trg_pos = kh_get_sha1(island_marks, trg_oid->hash);
> +     if (trg_pos >= kh_end(island_marks))
> +             return 1;
> +
> +     /*
> +      * if the source (our delta base) doesn't have a bitmap,
> +      * we don't want to base any deltas on it!
> +      */
> +     src_pos = kh_get_sha1(island_marks, src_oid->hash);
> +     if (src_pos >= kh_end(island_marks))
> +             return 0;
> +
> +     return island_bitmap_is_subset(kh_value(island_marks, trg_pos),
> +                             kh_value(island_marks, src_pos));
> +}
> +
> +int island_delta_cmp(const struct object_id *a, const struct object_id *b)
> +{
> +     khiter_t a_pos, b_pos;
> +     struct island_bitmap *a_bitmap = NULL, *b_bitmap = NULL;
> +
> +     if (!island_marks)
> +             return 0;
> +
> +     a_pos = kh_get_sha1(island_marks, a->hash);
> +     if (a_pos < kh_end(island_marks))
> +             a_bitmap = kh_value(island_marks, a_pos);
> +
> +     b_pos = kh_get_sha1(island_marks, b->hash);
> +     if (b_pos < kh_end(island_marks))
> +             b_bitmap = kh_value(island_marks, b_pos);
> +
> +     if (a_bitmap) {
> +             if (!b_bitmap || !island_bitmap_is_subset(a_bitmap, b_bitmap))
> +                     return -1;
> +     }
> +     if (b_bitmap) {
> +             if (!a_bitmap || !island_bitmap_is_subset(b_bitmap, a_bitmap))
> +                     return 1;
> +     }
> +
> +     return 0;
> +}
> +
> +static struct island_bitmap *create_or_get_island_marks(struct object *obj)
> +{
> +     khiter_t pos;
> +     int hash_ret;
> +
> +     pos = kh_put_sha1(island_marks, obj->oid.hash, &hash_ret);
> +     if (hash_ret)
> +             kh_value(island_marks, pos) = island_bitmap_new(NULL);
> +
> +     return kh_value(island_marks, pos);
> +}
> +
> +static void set_island_marks(struct object *obj, struct island_bitmap *marks)
> +{
> +     struct island_bitmap *b;
> +     khiter_t pos;
> +     int hash_ret;
> +
> +     pos = kh_put_sha1(island_marks, obj->oid.hash, &hash_ret);
> +     if (hash_ret) {
> +             /*
> +              * We don't have one yet; make a copy-on-write of the
> +              * parent.
> +              */
> +             marks->refcount++;
> +             kh_value(island_marks, pos) = marks;
> +             return;
> +     }
> +
> +     /*
> +      * We do have it. Make sure we split any copy-on-write before
> +      * updating.
> +      */
> +     b = kh_value(island_marks, pos);
> +     if (b->refcount > 1) {
> +             b->refcount--;
> +             b = kh_value(island_marks, pos) = island_bitmap_new(b);
> +     }
> +     island_bitmap_or(b, marks);
> +}
> +
> +static void mark_remote_island_1(struct remote_island *rl, int 
> is_core_island)
> +{
> +     uint32_t i;
> +
> +     for (i = 0; i < rl->oids.nr; ++i) {
> +             struct island_bitmap *marks;
> +             struct object *obj = parse_object(the_repository, 
> &rl->oids.oid[i]);
> +
> +             if (!obj)
> +                     continue;
> +
> +             marks = create_or_get_island_marks(obj);
> +             island_bitmap_set(marks, island_counter);
> +
> +             if (is_core_island && obj->type == OBJ_COMMIT)
> +                     obj->flags |= NEEDS_BITMAP;
> +
> +             /* If it was a tag, also make sure we hit the underlying 
> object. */
> +             while (obj && obj->type == OBJ_TAG) {
> +                     obj = ((struct tag *)obj)->tagged;
> +                     if (obj) {
> +                             parse_object(the_repository, &obj->oid);
> +                             marks = create_or_get_island_marks(obj);
> +                             island_bitmap_set(marks, island_counter);
> +                     }
> +             }
> +     }
> +
> +     if (is_core_island)
> +             island_counter_core = island_counter;
> +
> +     island_counter++;
> +}
> +
> +static int cmp_tree_depth(const void *va, const void *vb)
> +{
> +     struct object_entry *a = *(struct object_entry **)va;
> +     struct object_entry *b = *(struct object_entry **)vb;
> +     return a->tree_depth - b->tree_depth;
> +}
> +
> +void resolve_tree_islands(int progress, struct packing_data *to_pack)
> +{
> +     struct progress *progress_state = NULL;
> +     struct object_entry **todo;
> +     int nr = 0;
> +     int i;
> +
> +     if (!island_marks)
> +             return;
> +
> +     /*
> +      * We process only trees, as commits and tags have already been handled
> +      * (and passed their marks on to root trees, as well. We must make sure
> +      * to process them in descending tree-depth order so that marks
> +      * propagate down the tree properly, even if a sub-tree is found in
> +      * multiple parent trees.
> +      */
> +     ALLOC_ARRAY(todo, to_pack->nr_objects);
> +     for (i = 0; i < to_pack->nr_objects; i++) {
> +             if (oe_type(&to_pack->objects[i]) == OBJ_TREE)
> +                     todo[nr++] = &to_pack->objects[i];
> +     }
> +     QSORT(todo, nr, cmp_tree_depth);
> +
> +     if (progress)
> +             progress_state = start_progress(_("Propagating island marks"), 
> nr);
> +
> +     for (i = 0; i < nr; i++) {
> +             struct object_entry *ent = todo[i];
> +             struct island_bitmap *root_marks;
> +             struct tree *tree;
> +             struct tree_desc desc;
> +             struct name_entry entry;
> +             khiter_t pos;
> +
> +             pos = kh_get_sha1(island_marks, ent->idx.oid.hash);
> +             if (pos >= kh_end(island_marks))
> +                     continue;
> +
> +             root_marks = kh_value(island_marks, pos);
> +
> +             tree = lookup_tree(the_repository, &ent->idx.oid);
> +             if (!tree || parse_tree(tree) < 0)
> +                     die(_("bad tree object %s"), oid_to_hex(&ent->idx.oid));
> +
> +             init_tree_desc(&desc, tree->buffer, tree->size);
> +             while (tree_entry(&desc, &entry)) {
> +                     struct object *obj;
> +
> +                     if (S_ISGITLINK(entry.mode))
> +                             continue;
> +
> +                     obj = lookup_object(the_repository, entry.oid->hash);
> +                     if (!obj)
> +                             continue;
> +
> +                     set_island_marks(obj, root_marks);
> +             }
> +
> +             free_tree_buffer(tree);
> +
> +             display_progress(progress_state, i+1);
> +     }
> +
> +     stop_progress(&progress_state);
> +     free(todo);
> +}
> +
> +static regex_t *island_regexes;
> +static unsigned int island_regexes_alloc, island_regexes_nr;
> +static const char *core_island_name;
> +
> +static int island_config_callback(const char *k, const char *v, void *cb)
> +{
> +     if (!strcmp(k, "pack.island")) {
> +             struct strbuf re = STRBUF_INIT;
> +
> +             if (!v)
> +                     return config_error_nonbool(k);
> +
> +             ALLOC_GROW(island_regexes, island_regexes_nr + 1, 
> island_regexes_alloc);
> +
> +             if (*v != '^')
> +                     strbuf_addch(&re, '^');
> +             strbuf_addstr(&re, v);
> +
> +             if (regcomp(&island_regexes[island_regexes_nr], re.buf, 
> REG_EXTENDED))
> +                     die(_("failed to load island regex for '%s': %s"), k, 
> re.buf);
> +
> +             strbuf_release(&re);
> +             island_regexes_nr++;
> +             return 0;
> +     }
> +
> +     if (!strcmp(k, "pack.islandcore"))
> +             return git_config_string(&core_island_name, k, v);
> +
> +     return 0;
> +}
> +
> +static void add_ref_to_island(const char *island_name, const struct 
> object_id *oid)
> +{
> +     uint64_t sha_core;
> +     struct remote_island *rl = NULL;
> +
> +     int hash_ret;
> +     khiter_t pos = kh_put_str(remote_islands, island_name, &hash_ret);
> +
> +     if (hash_ret) {
> +             kh_key(remote_islands, pos) = xstrdup(island_name);
> +             kh_value(remote_islands, pos) = xcalloc(1, sizeof(struct 
> remote_island));
> +     }
> +
> +     rl = kh_value(remote_islands, pos);
> +     oid_array_append(&rl->oids, oid);
> +
> +     memcpy(&sha_core, oid->hash, sizeof(uint64_t));
> +     rl->hash += sha_core;

Hmm, so the first 64-bits of the oid of each ref that is part of
this island is added together as a 'hash' for the island. And this
is used to de-duplicate the islands? Any false positives? (does it
matter - it would only affect performance, not correctness, right?)

> +}
> +
> +static int find_island_for_ref(const char *refname, const struct object_id 
> *oid,
> +                            int flags, void *data)
> +{
> +     /*
> +      * We should advertise 'ARRAY_SIZE(matches) - 2' as the max,
> +      * so we can diagnose below a config with more capture groups
> +      * than we support.
> +      */
> +     regmatch_t matches[16];
> +     int i, m;
> +     struct strbuf island_name = STRBUF_INIT;
> +
> +     /* walk backwards to get last-one-wins ordering */
> +     for (i = island_regexes_nr - 1; i >= 0; i--) {
> +             if (!regexec(&island_regexes[i], refname,
> +                          ARRAY_SIZE(matches), matches, 0))
> +                     break;
> +     }
> +
> +     if (i < 0)
> +             return 0;
> +
> +     if (matches[ARRAY_SIZE(matches) - 1].rm_so != -1)
> +             warning(_("island regex from config has "
> +                       "too many capture groups (max=%d)"),
> +                     (int)ARRAY_SIZE(matches) - 2);
> +
> +     for (m = 1; m < ARRAY_SIZE(matches); m++) {
> +             regmatch_t *match = &matches[m];
> +
> +             if (match->rm_so == -1)
> +                     continue;
> +
> +             if (island_name.len)
> +                     strbuf_addch(&island_name, '-');
> +
> +             strbuf_add(&island_name, refname + match->rm_so, match->rm_eo - 
> match->rm_so);
> +     }
> +
> +     add_ref_to_island(island_name.buf, oid);
> +     strbuf_release(&island_name);
> +     return 0;
> +}
> +
> +static struct remote_island *get_core_island(void)
> +{
> +     if (core_island_name) {
> +             khiter_t pos = kh_get_str(remote_islands, core_island_name);
> +             if (pos < kh_end(remote_islands))
> +                     return kh_value(remote_islands, pos);
> +     }
> +
> +     return NULL;
> +}
> +
> +static void deduplicate_islands(void)
> +{
> +     struct remote_island *island, *core = NULL, **list;
> +     unsigned int island_count, dst, src, ref, i = 0;
> +
> +     island_count = kh_size(remote_islands);
> +     ALLOC_ARRAY(list, island_count);
> +
> +     kh_foreach_value(remote_islands, island, {
> +             list[i++] = island;
> +     });
> +
> +     for (ref = 0; ref + 1 < island_count; ref++) {
> +             for (src = ref + 1, dst = src; src < island_count; src++) {
> +                     if (list[ref]->hash == list[src]->hash)
> +                             continue;
> +
> +                     if (src != dst)
> +                             list[dst] = list[src];
> +
> +                     dst++;
> +             }
> +             island_count = dst;
> +     }
> +
> +     island_bitmap_size = (island_count / 32) + 1;
> +     core = get_core_island();
> +
> +     for (i = 0; i < island_count; ++i) {
> +             mark_remote_island_1(list[i], core && list[i]->hash == 
> core->hash);
> +     }
> +
> +     free(list);
> +}
> +
> +void load_delta_islands(void)
> +{
> +     island_marks = kh_init_sha1();
> +     remote_islands = kh_init_str();
> +
> +     git_config(island_config_callback, NULL);
> +     for_each_ref(find_island_for_ref, NULL);
> +     deduplicate_islands();
> +
> +     fprintf(stderr, _("Marked %d islands, done.\n"), island_counter);
> +}
> +
> +void propagate_island_marks(struct commit *commit)
> +{
> +     khiter_t pos = kh_get_sha1(island_marks, commit->object.oid.hash);
> +
> +     if (pos < kh_end(island_marks)) {
> +             struct commit_list *p;
> +             struct island_bitmap *root_marks = kh_value(island_marks, pos);
> +
> +             parse_commit(commit);
> +             set_island_marks(&get_commit_tree(commit)->object, root_marks);
> +             for (p = commit->parents; p; p = p->next)
> +                     set_island_marks(&p->item->object, root_marks);
> +     }
> +}
> +
> +int compute_pack_layers(struct packing_data *to_pack)
> +{
> +     uint32_t i;
> +
> +     if (!core_island_name || !island_marks)
> +             return 1;
> +
> +     for (i = 0; i < to_pack->nr_objects; ++i) {
> +             struct object_entry *entry = &to_pack->objects[i];
> +             khiter_t pos = kh_get_sha1(island_marks, entry->idx.oid.hash);
> +
> +             entry->layer = 1;
> +
> +             if (pos < kh_end(island_marks)) {
> +                     struct island_bitmap *bitmap = kh_value(island_marks, 
> pos);
> +
> +                     if (island_bitmap_get(bitmap, island_counter_core))
> +                             entry->layer = 0;
> +             }
> +     }
> +
> +     return 2;
> +}
> diff --git a/delta-islands.h b/delta-islands.h
> new file mode 100644
> index 0000000000..f9725730f4
> --- /dev/null
> +++ b/delta-islands.h
> @@ -0,0 +1,11 @@
> +#ifndef DELTA_ISLANDS_H
> +#define DELTA_ISLANDS_H
> +
> +int island_delta_cmp(const struct object_id *a, const struct object_id *b);
> +int in_same_island(const struct object_id *, const struct object_id *);
> +void resolve_tree_islands(int progress, struct packing_data *to_pack);
> +void load_delta_islands(void);
> +void propagate_island_marks(struct commit *commit);
> +int compute_pack_layers(struct packing_data *to_pack);
> +
> +#endif /* DELTA_ISLANDS_H */
> diff --git a/pack-objects.h b/pack-objects.h
> index edf74dabdd..8eecd67991 100644
> --- a/pack-objects.h
> +++ b/pack-objects.h
> @@ -100,6 +100,10 @@ struct object_entry {
>       unsigned type_:TYPE_BITS;
>       unsigned no_try_delta:1;
>       unsigned in_pack_type:TYPE_BITS; /* could be delta */
> +
> +     unsigned int tree_depth; /* should be repositioned for packing? */
> +     unsigned char layer;
> +
>       unsigned preferred_base:1; /*
>                                   * we do not pack this, but is available
>                                   * to be used as the base object to delta
> 

Sorry, I spent so long reading this patch, I have run out of
time tonight (and I am busy tomorrow) to read the rest of the
series.

ATB,
Ramsay Jones

Reply via email to