object.con commit diff-tree: support list if input trees on stdin (e0965d8)
   1#include "object.h"
   2#include "blob.h"
   3#include "tree.h"
   4#include "commit.h"
   5#include "cache.h"
   6#include "tag.h"
   7#include <stdlib.h>
   8#include <string.h>
   9
  10struct object **objs;
  11int nr_objs;
  12static int obj_allocs;
  13
  14static int find_object(unsigned char *sha1)
  15{
  16        int first = 0, last = nr_objs;
  17
  18        while (first < last) {
  19                int next = (first + last) / 2;
  20                struct object *obj = objs[next];
  21                int cmp;
  22
  23                cmp = memcmp(sha1, obj->sha1, 20);
  24                if (!cmp)
  25                        return next;
  26                if (cmp < 0) {
  27                        last = next;
  28                        continue;
  29                }
  30                first = next+1;
  31        }
  32        return -first-1;
  33}
  34
  35struct object *lookup_object(unsigned char *sha1)
  36{
  37        int pos = find_object(sha1);
  38        if (pos >= 0)
  39                return objs[pos];
  40        return NULL;
  41}
  42
  43void created_object(unsigned char *sha1, struct object *obj)
  44{
  45        int pos = find_object(sha1);
  46
  47        obj->parsed = 0;
  48        memcpy(obj->sha1, sha1, 20);
  49        obj->type = NULL;
  50        obj->refs = NULL;
  51        obj->used = 0;
  52
  53        if (pos >= 0)
  54                die("Inserting %s twice\n", sha1_to_hex(sha1));
  55        pos = -pos-1;
  56
  57        if (obj_allocs == nr_objs) {
  58                obj_allocs = alloc_nr(obj_allocs);
  59                objs = xrealloc(objs, obj_allocs * sizeof(struct object *));
  60        }
  61
  62        /* Insert it into the right place */
  63        memmove(objs + pos + 1, objs + pos, (nr_objs - pos) * 
  64                sizeof(struct object *));
  65
  66        objs[pos] = obj;
  67        nr_objs++;
  68}
  69
  70void add_ref(struct object *refer, struct object *target)
  71{
  72        struct object_list **pp = &refer->refs;
  73        struct object_list *p;
  74        
  75        while ((p = *pp) != NULL) {
  76                if (p->item == target)
  77                        return;
  78                pp = &p->next;
  79        }
  80
  81        target->used = 1;
  82        p = xmalloc(sizeof(*p));
  83        p->item = target;
  84        p->next = NULL;
  85        *pp = p;
  86}
  87
  88void mark_reachable(struct object *obj, unsigned int mask)
  89{
  90        struct object_list *p = obj->refs;
  91
  92        /* If we've been here already, don't bother */
  93        if (obj->flags & mask)
  94                return;
  95        obj->flags |= mask;
  96        while (p) {
  97                mark_reachable(p->item, mask);
  98                p = p->next;
  99        }
 100}
 101
 102struct object *parse_object(unsigned char *sha1)
 103{
 104        unsigned long mapsize;
 105        void *map = map_sha1_file(sha1, &mapsize);
 106        if (map) {
 107                char type[100];
 108                unsigned long size;
 109                void *buffer = unpack_sha1_file(map, mapsize, type, &size);
 110                munmap(map, mapsize);
 111                if (!buffer)
 112                        return NULL;
 113                if (check_sha1_signature(sha1, buffer, size, type) < 0)
 114                        printf("sha1 mismatch %s\n", sha1_to_hex(sha1));
 115                free(buffer);
 116                if (!strcmp(type, "blob")) {
 117                        struct blob *ret = lookup_blob(sha1);
 118                        parse_blob(ret);
 119                        return &ret->object;
 120                } else if (!strcmp(type, "tree")) {
 121                        struct tree *ret = lookup_tree(sha1);
 122                        parse_tree(ret);
 123                        return &ret->object;
 124                } else if (!strcmp(type, "commit")) {
 125                        struct commit *ret = lookup_commit(sha1);
 126                        parse_commit(ret);
 127                        return &ret->object;
 128                } else if (!strcmp(type, "tag")) {
 129                        struct tag *ret = lookup_tag(sha1);
 130                        parse_tag(ret);
 131                        return &ret->object;
 132                } else {
 133                        return NULL;
 134                }
 135        }
 136        return NULL;
 137}