rev-cache.con commit request-pull: minor tweaks. (9969b64)
   1#include "refs.h"
   2#include "cache.h"
   3#include "rev-cache.h"
   4
   5struct rev_cache **rev_cache;
   6int nr_revs, alloc_revs;
   7
   8struct rev_list_elem *rle_free;
   9
  10#define BATCH_SIZE 512
  11
  12int find_rev_cache(const unsigned char *sha1)
  13{
  14        int lo = 0, hi = nr_revs;
  15        while (lo < hi) {
  16                int mi = (lo + hi) / 2;
  17                struct rev_cache *ri = rev_cache[mi];
  18                int cmp = memcmp(sha1, ri->sha1, 20);
  19                if (!cmp)
  20                        return mi;
  21                if (cmp < 0)
  22                        hi = mi;
  23                else
  24                        lo = mi + 1;
  25        }
  26        return -lo - 1;
  27}
  28
  29static struct rev_list_elem *alloc_list_elem(void)
  30{
  31        struct rev_list_elem *rle;
  32        if (!rle_free) {
  33                int i;
  34
  35                rle = xmalloc(sizeof(*rle) * BATCH_SIZE);
  36                for (i = 0; i < BATCH_SIZE - 1; i++) {
  37                        rle[i].ri = NULL;
  38                        rle[i].next = &rle[i + 1];
  39                }
  40                rle[BATCH_SIZE - 1].ri = NULL;
  41                rle[BATCH_SIZE - 1].next = NULL;
  42                rle_free = rle;
  43        }
  44        rle = rle_free;
  45        rle_free = rle->next;
  46        return rle;
  47}
  48
  49static struct rev_cache *create_rev_cache(const unsigned char *sha1)
  50{
  51        struct rev_cache *ri;
  52        int pos = find_rev_cache(sha1);
  53
  54        if (0 <= pos)
  55                return rev_cache[pos];
  56        pos = -pos - 1;
  57        if (alloc_revs <= ++nr_revs) {
  58                alloc_revs = alloc_nr(alloc_revs);
  59                rev_cache = xrealloc(rev_cache, sizeof(ri) * alloc_revs);
  60        }
  61        if (pos < nr_revs)
  62                memmove(rev_cache + pos + 1, rev_cache + pos,
  63                        (nr_revs - pos - 1) * sizeof(ri));
  64        ri = xcalloc(1, sizeof(*ri));
  65        memcpy(ri->sha1, sha1, 20);
  66        rev_cache[pos] = ri;
  67        return ri;
  68}
  69
  70static unsigned char last_sha1[20];
  71
  72static void write_one_rev_cache(FILE *rev_cache_file, struct rev_cache *ri)
  73{
  74        unsigned char flag;
  75        struct rev_list_elem *rle;
  76
  77        if (ri->written)
  78                return;
  79
  80        if (ri->parsed) {
  81                /* We use last_sha1 compression only for the first parent;
  82                 * otherwise the resulting rev-cache would lose the parent
  83                 * order information.
  84                 */
  85                if (ri->parents &&
  86                    !memcmp(ri->parents->ri->sha1, last_sha1, 20))
  87                        flag = (ri->num_parents - 1) | 0x80;
  88                else
  89                        flag = ri->num_parents;
  90
  91                fwrite(ri->sha1, 20, 1, rev_cache_file);
  92                fwrite(&flag, 1, 1, rev_cache_file);
  93                for (rle = ri->parents; rle; rle = rle->next) {
  94                        if (flag & 0x80 && rle == ri->parents)
  95                                continue;
  96                        fwrite(rle->ri->sha1, 20, 1, rev_cache_file);
  97                }
  98                memcpy(last_sha1, ri->sha1, 20);
  99                ri->written = 1;
 100        }
 101        /* recursively write children depth first */
 102        for (rle = ri->children; rle; rle = rle->next)
 103                write_one_rev_cache(rev_cache_file, rle->ri);
 104}
 105
 106void write_rev_cache(const char *newpath, const char *oldpath)
 107{
 108        /* write the following commit ancestry information in
 109         * $GIT_DIR/info/rev-cache.
 110         *
 111         * The format is:
 112         * 20-byte SHA1 (commit ID)
 113         * 1-byte flag:
 114         * - bit 0-6 records "number of parent commit SHA1s to
 115         *   follow" (i.e. up to 127 children can be listed).
 116         * - when the bit 7 is on, then "the entry immediately
 117         *   before this entry is one of the parents of this
 118         *   commit".
 119         * N x 20-byte SHA1 (parent commit IDs)
 120         */
 121        FILE *rev_cache_file;
 122        int i;
 123        struct rev_cache *ri;
 124
 125        if (!strcmp(newpath, oldpath)) {
 126                /* If we are doing it in place */
 127                rev_cache_file = fopen(newpath, "a");
 128        }
 129        else {
 130                char buf[8096];
 131                size_t sz;
 132                FILE *oldfp = fopen(oldpath, "r");
 133                rev_cache_file = fopen(newpath, "w");
 134                if (oldfp) {
 135                        while (1) {
 136                                sz = fread(buf, 1, sizeof(buf), oldfp);
 137                                if (sz == 0)
 138                                        break;
 139                                fwrite(buf, 1, sz, rev_cache_file);
 140                        }
 141                        fclose(oldfp);
 142                }
 143        }
 144
 145        memset(last_sha1, 0, 20);
 146
 147        /* Go through available rev_cache structures, starting from
 148         * parentless ones first, so that we would get most out of
 149         * last_sha1 optimization by the depth first behaviour of
 150         * write_one_rev_cache().
 151         */
 152        for (i = 0; i < nr_revs; i++) {
 153                ri = rev_cache[i];
 154                if (ri->num_parents)
 155                        continue;
 156                write_one_rev_cache(rev_cache_file, ri);
 157        }
 158        /* Then the rest */
 159        for (i = 0; i < nr_revs; i++) {
 160                ri = rev_cache[i];
 161                write_one_rev_cache(rev_cache_file, ri);
 162        }
 163        fclose(rev_cache_file);
 164}
 165
 166static void add_parent(struct rev_cache *child,
 167                       const unsigned char *parent_sha1)
 168{
 169        struct rev_cache *parent = create_rev_cache(parent_sha1);
 170        struct rev_list_elem *e = alloc_list_elem();
 171
 172        /* Keep the parent list ordered in the same way the commit
 173         * object records them.
 174         */
 175        e->ri = parent;
 176        e->next = NULL;
 177        if (!child->parents_tail)
 178                child->parents = e;
 179        else
 180                child->parents_tail->next = e;
 181        child->parents_tail = e;
 182        child->num_parents++;
 183
 184        /* There is no inherent order of the children so we just
 185         * LIFO them together.
 186         */
 187        e = alloc_list_elem();
 188        e->next = parent->children;
 189        parent->children = e;
 190        e->ri = child;
 191        parent->num_children++;
 192}
 193
 194int read_rev_cache(const char *path, FILE *dumpfile, int dry_run)
 195{
 196        unsigned char *map;
 197        int fd;
 198        struct stat st;
 199        unsigned long ofs, len;
 200        struct rev_cache *ri = NULL;
 201
 202        fd = open(path, O_RDONLY);
 203        if (fd < 0) {
 204                if (dry_run)
 205                        return error("cannot open %s", path);
 206                if (errno == ENOENT)
 207                        return 0;
 208                return -1;
 209        }
 210        if (fstat(fd, &st)) {
 211                close(fd);
 212                return -1;
 213        }
 214        map = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
 215        if (map == MAP_FAILED) {
 216                close(fd);
 217                return -1;
 218        }
 219        close(fd);
 220
 221        memset(last_sha1, 0, 20);
 222        ofs = 0;
 223        len = st.st_size;
 224        while (ofs < len) {
 225                unsigned char sha1[20];
 226                int flag, cnt, i;
 227                if (len < ofs + 21)
 228                        die("rev-cache too short");
 229                memcpy(sha1, map + ofs, 20);
 230                flag = map[ofs + 20];
 231                ofs += 21;
 232                cnt = (flag & 0x7f) + ((flag & 0x80) != 0);
 233                if (len < ofs + (flag & 0x7f) * 20)
 234                        die("rev-cache too short to have %d more parents",
 235                            (flag & 0x7f));
 236                if (dumpfile)
 237                        fprintf(dumpfile, "%s", sha1_to_hex(sha1));
 238                if (!dry_run) {
 239                        ri = create_rev_cache(sha1);
 240                        if (!ri)
 241                                die("cannot create rev-cache for %s",
 242                                    sha1_to_hex(sha1));
 243                        ri->written = ri->parsed = 1;
 244                }
 245                i = 0;
 246                if (flag & 0x80) {
 247                        if (!dry_run)
 248                                add_parent(ri, last_sha1);
 249                        if (dumpfile)
 250                                fprintf(dumpfile, " %s",
 251                                        sha1_to_hex(last_sha1));
 252                        i++;
 253                }
 254                while (i++ < cnt) {
 255                        if (!dry_run)
 256                                add_parent(ri, map + ofs);
 257                        if (dumpfile)
 258                                fprintf(dumpfile, " %s",
 259                                        sha1_to_hex(last_sha1));
 260                        ofs += 20;
 261                }
 262                if (dumpfile)
 263                        fprintf(dumpfile, "\n");
 264                memcpy(last_sha1, sha1, 20);
 265        }
 266        if (ofs != len)
 267                die("rev-cache truncated?");
 268        munmap(map, len);
 269        return 0;
 270}
 271
 272int record_rev_cache(const unsigned char *sha1, FILE *dumpfile)
 273{
 274        unsigned char parent[20];
 275        char type[20];
 276        unsigned long size, ofs;
 277        unsigned int cnt, i;
 278        void *buf;
 279        struct rev_cache *ri;
 280
 281        buf = read_sha1_file(sha1, type, &size);
 282        if (!buf)
 283                return error("%s: not found", sha1_to_hex(sha1));
 284        if (strcmp(type, "commit")) {
 285                free(buf);
 286                return error("%s: not a commit but a %s",
 287                             sha1_to_hex(sha1), type);
 288        }
 289        ri = create_rev_cache(sha1);
 290        if (ri->parsed)
 291                return 0;
 292        if (dumpfile)
 293                fprintf(dumpfile, "commit %s\n", sha1_to_hex(sha1));
 294
 295        cnt = 0;
 296        ofs = 46; /* "tree " + hex-sha1 + "\n" */
 297        while (!memcmp(buf + ofs, "parent ", 7) &&
 298               !get_sha1_hex(buf + ofs + 7, parent)) {
 299                ofs += 48;
 300                cnt++;
 301        }
 302        if (cnt * 48 + 46 != ofs) {
 303                free(buf);
 304                die("internal error in record_rev_cache");
 305        }
 306
 307        ri = create_rev_cache(sha1);
 308        ri->parsed = 1;
 309
 310        for (i = 0; i < cnt; i++) {
 311                unsigned char parent_sha1[20];
 312
 313                ofs = 46 + i * 48 + 7;
 314                get_sha1_hex(buf + ofs, parent_sha1);
 315                add_parent(ri, parent_sha1);
 316                record_rev_cache(parent_sha1, dumpfile);
 317        }
 318        free(buf);
 319        return 0;
 320}