2  * Copyright (c) 2011, Google Inc.
 
  14 typedef int (*open_istream_fn)(struct git_istream *,
 
  16                                const unsigned char *,
 
  18 typedef int (*close_istream_fn)(struct git_istream *);
 
  19 typedef ssize_t (*read_istream_fn)(struct git_istream *, char *, size_t);
 
  22         close_istream_fn close;
 
  26 #define open_method_decl(name) \
 
  27         int open_istream_ ##name \
 
  28         (struct git_istream *st, struct object_info *oi, \
 
  29          const unsigned char *sha1, \
 
  30          enum object_type *type)
 
  32 #define close_method_decl(name) \
 
  33         int close_istream_ ##name \
 
  34         (struct git_istream *st)
 
  36 #define read_method_decl(name) \
 
  37         ssize_t read_istream_ ##name \
 
  38         (struct git_istream *st, char *buf, size_t sz)
 
  40 /* forward declaration */
 
  41 static open_method_decl(incore);
 
  42 static open_method_decl(loose);
 
  43 static open_method_decl(pack_non_delta);
 
  44 static struct git_istream *attach_stream_filter(struct git_istream *st,
 
  45                                                 struct stream_filter *filter);
 
  48 static open_istream_fn open_istream_tbl[] = {
 
  51         open_istream_pack_non_delta,
 
  54 #define FILTER_BUFFER (1024*16)
 
  56 struct filtered_istream {
 
  57         struct git_istream *upstream;
 
  58         struct stream_filter *filter;
 
  59         char ibuf[FILTER_BUFFER];
 
  60         char obuf[FILTER_BUFFER];
 
  67         const struct stream_vtbl *vtbl;
 
  68         unsigned long size; /* inflated size of full object */
 
  70         enum { z_unused, z_used, z_done, z_error } z_state;
 
  74                         char *buf; /* from read_object() */
 
  75                         unsigned long read_ptr;
 
  80                         unsigned long mapsize;
 
  87                         struct packed_git *pack;
 
  91                 struct filtered_istream filtered;
 
  95 int close_istream(struct git_istream *st)
 
  97         int r = st->vtbl->close(st);
 
 102 ssize_t read_istream(struct git_istream *st, void *buf, size_t sz)
 
 104         return st->vtbl->read(st, buf, sz);
 
 107 static enum input_source istream_source(const unsigned char *sha1,
 
 108                                         enum object_type *type,
 
 109                                         struct object_info *oi)
 
 116         status = sha1_object_info_extended(sha1, oi, 0);
 
 120         switch (oi->whence) {
 
 124                 if (!oi->u.packed.is_delta && big_file_threshold < size)
 
 125                         return pack_non_delta;
 
 132 struct git_istream *open_istream(const unsigned char *sha1,
 
 133                                  enum object_type *type,
 
 135                                  struct stream_filter *filter)
 
 137         struct git_istream *st;
 
 138         struct object_info oi = {NULL};
 
 139         const unsigned char *real = lookup_replace_object(sha1);
 
 140         enum input_source src = istream_source(real, type, &oi);
 
 145         st = xmalloc(sizeof(*st));
 
 146         if (open_istream_tbl[src](st, &oi, real, type)) {
 
 147                 if (open_istream_incore(st, &oi, real, type)) {
 
 153                 /* Add "&& !is_null_stream_filter(filter)" for performance */
 
 154                 struct git_istream *nst = attach_stream_filter(st, filter);
 
 165 /*****************************************************************
 
 169  *****************************************************************/
 
 171 static void close_deflated_stream(struct git_istream *st)
 
 173         if (st->z_state == z_used)
 
 174                 git_inflate_end(&st->z);
 
 178 /*****************************************************************
 
 182  *****************************************************************/
 
 184 static close_method_decl(filtered)
 
 186         free_stream_filter(st->u.filtered.filter);
 
 187         return close_istream(st->u.filtered.upstream);
 
 190 static read_method_decl(filtered)
 
 192         struct filtered_istream *fs = &(st->u.filtered);
 
 196                 /* do we already have filtered output? */
 
 197                 if (fs->o_ptr < fs->o_end) {
 
 198                         size_t to_move = fs->o_end - fs->o_ptr;
 
 201                         memcpy(buf + filled, fs->obuf + fs->o_ptr, to_move);
 
 202                         fs->o_ptr += to_move;
 
 207                 fs->o_end = fs->o_ptr = 0;
 
 209                 /* do we have anything to feed the filter with? */
 
 210                 if (fs->i_ptr < fs->i_end) {
 
 211                         size_t to_feed = fs->i_end - fs->i_ptr;
 
 212                         size_t to_receive = FILTER_BUFFER;
 
 213                         if (stream_filter(fs->filter,
 
 214                                           fs->ibuf + fs->i_ptr, &to_feed,
 
 215                                           fs->obuf, &to_receive))
 
 217                         fs->i_ptr = fs->i_end - to_feed;
 
 218                         fs->o_end = FILTER_BUFFER - to_receive;
 
 222                 /* tell the filter to drain upon no more input */
 
 223                 if (fs->input_finished) {
 
 224                         size_t to_receive = FILTER_BUFFER;
 
 225                         if (stream_filter(fs->filter,
 
 227                                           fs->obuf, &to_receive))
 
 229                         fs->o_end = FILTER_BUFFER - to_receive;
 
 234                 fs->i_end = fs->i_ptr = 0;
 
 236                 /* refill the input from the upstream */
 
 237                 if (!fs->input_finished) {
 
 238                         fs->i_end = read_istream(fs->upstream, fs->ibuf, FILTER_BUFFER);
 
 244                 fs->input_finished = 1;
 
 249 static struct stream_vtbl filtered_vtbl = {
 
 250         close_istream_filtered,
 
 251         read_istream_filtered,
 
 254 static struct git_istream *attach_stream_filter(struct git_istream *st,
 
 255                                                 struct stream_filter *filter)
 
 257         struct git_istream *ifs = xmalloc(sizeof(*ifs));
 
 258         struct filtered_istream *fs = &(ifs->u.filtered);
 
 260         ifs->vtbl = &filtered_vtbl;
 
 263         fs->i_end = fs->i_ptr = 0;
 
 264         fs->o_end = fs->o_ptr = 0;
 
 265         fs->input_finished = 0;
 
 266         ifs->size = -1; /* unknown */
 
 270 /*****************************************************************
 
 272  * Loose object stream
 
 274  *****************************************************************/
 
 276 static read_method_decl(loose)
 
 278         size_t total_read = 0;
 
 280         switch (st->z_state) {
 
 289         if (st->u.loose.hdr_used < st->u.loose.hdr_avail) {
 
 290                 size_t to_copy = st->u.loose.hdr_avail - st->u.loose.hdr_used;
 
 293                 memcpy(buf, st->u.loose.hdr + st->u.loose.hdr_used, to_copy);
 
 294                 st->u.loose.hdr_used += to_copy;
 
 295                 total_read += to_copy;
 
 298         while (total_read < sz) {
 
 301                 st->z.next_out = (unsigned char *)buf + total_read;
 
 302                 st->z.avail_out = sz - total_read;
 
 303                 status = git_inflate(&st->z, Z_FINISH);
 
 305                 total_read = st->z.next_out - (unsigned char *)buf;
 
 307                 if (status == Z_STREAM_END) {
 
 308                         git_inflate_end(&st->z);
 
 309                         st->z_state = z_done;
 
 312                 if (status != Z_OK && (status != Z_BUF_ERROR || total_read < sz)) {
 
 313                         git_inflate_end(&st->z);
 
 314                         st->z_state = z_error;
 
 321 static close_method_decl(loose)
 
 323         close_deflated_stream(st);
 
 324         munmap(st->u.loose.mapped, st->u.loose.mapsize);
 
 328 static struct stream_vtbl loose_vtbl = {
 
 333 static open_method_decl(loose)
 
 335         st->u.loose.mapped = map_sha1_file(sha1, &st->u.loose.mapsize);
 
 336         if (!st->u.loose.mapped)
 
 338         if (unpack_sha1_header(&st->z,
 
 342                                sizeof(st->u.loose.hdr)) < 0) {
 
 343                 git_inflate_end(&st->z);
 
 344                 munmap(st->u.loose.mapped, st->u.loose.mapsize);
 
 348         parse_sha1_header(st->u.loose.hdr, &st->size);
 
 349         st->u.loose.hdr_used = strlen(st->u.loose.hdr) + 1;
 
 350         st->u.loose.hdr_avail = st->z.total_out;
 
 351         st->z_state = z_used;
 
 353         st->vtbl = &loose_vtbl;
 
 358 /*****************************************************************
 
 360  * Non-delta packed object stream
 
 362  *****************************************************************/
 
 364 static read_method_decl(pack_non_delta)
 
 366         size_t total_read = 0;
 
 368         switch (st->z_state) {
 
 370                 memset(&st->z, 0, sizeof(st->z));
 
 371                 git_inflate_init(&st->z);
 
 372                 st->z_state = z_used;
 
 382         while (total_read < sz) {
 
 384                 struct pack_window *window = NULL;
 
 385                 unsigned char *mapped;
 
 387                 mapped = use_pack(st->u.in_pack.pack, &window,
 
 388                                   st->u.in_pack.pos, &st->z.avail_in);
 
 390                 st->z.next_out = (unsigned char *)buf + total_read;
 
 391                 st->z.avail_out = sz - total_read;
 
 392                 st->z.next_in = mapped;
 
 393                 status = git_inflate(&st->z, Z_FINISH);
 
 395                 st->u.in_pack.pos += st->z.next_in - mapped;
 
 396                 total_read = st->z.next_out - (unsigned char *)buf;
 
 399                 if (status == Z_STREAM_END) {
 
 400                         git_inflate_end(&st->z);
 
 401                         st->z_state = z_done;
 
 404                 if (status != Z_OK && status != Z_BUF_ERROR) {
 
 405                         git_inflate_end(&st->z);
 
 406                         st->z_state = z_error;
 
 413 static close_method_decl(pack_non_delta)
 
 415         close_deflated_stream(st);
 
 419 static struct stream_vtbl pack_non_delta_vtbl = {
 
 420         close_istream_pack_non_delta,
 
 421         read_istream_pack_non_delta,
 
 424 static open_method_decl(pack_non_delta)
 
 426         struct pack_window *window;
 
 427         enum object_type in_pack_type;
 
 429         st->u.in_pack.pack = oi->u.packed.pack;
 
 430         st->u.in_pack.pos = oi->u.packed.offset;
 
 433         in_pack_type = unpack_object_header(st->u.in_pack.pack,
 
 438         switch (in_pack_type) {
 
 440                 return -1; /* we do not do deltas for now */
 
 447         st->z_state = z_unused;
 
 448         st->vtbl = &pack_non_delta_vtbl;
 
 453 /*****************************************************************
 
 457  *****************************************************************/
 
 459 static close_method_decl(incore)
 
 461         free(st->u.incore.buf);
 
 465 static read_method_decl(incore)
 
 467         size_t read_size = sz;
 
 468         size_t remainder = st->size - st->u.incore.read_ptr;
 
 470         if (remainder <= read_size)
 
 471                 read_size = remainder;
 
 473                 memcpy(buf, st->u.incore.buf + st->u.incore.read_ptr, read_size);
 
 474                 st->u.incore.read_ptr += read_size;
 
 479 static struct stream_vtbl incore_vtbl = {
 
 480         close_istream_incore,
 
 484 static open_method_decl(incore)
 
 486         st->u.incore.buf = read_sha1_file_extended(sha1, type, &st->size, 0);
 
 487         st->u.incore.read_ptr = 0;
 
 488         st->vtbl = &incore_vtbl;
 
 490         return st->u.incore.buf ? 0 : -1;
 
 494 /****************************************************************
 
 495  * Users of streaming interface
 
 496  ****************************************************************/
 
 498 int stream_blob_to_fd(int fd, unsigned const char *sha1, struct stream_filter *filter,
 
 501         struct git_istream *st;
 
 502         enum object_type type;
 
 507         st = open_istream(sha1, &type, &sz, filter);
 
 510         if (type != OBJ_BLOB)
 
 514                 ssize_t wrote, holeto;
 
 515                 ssize_t readlen = read_istream(st, buf, sizeof(buf));
 
 521                 if (can_seek && sizeof(buf) == readlen) {
 
 522                         for (holeto = 0; holeto < readlen; holeto++)
 
 525                         if (readlen == holeto) {
 
 531                 if (kept && lseek(fd, kept, SEEK_CUR) == (off_t) -1)
 
 535                 wrote = write_in_full(fd, buf, readlen);
 
 537                 if (wrote != readlen)
 
 540         if (kept && (lseek(fd, kept - 1, SEEK_CUR) == (off_t) -1 ||
 
 541                      xwrite(fd, "", 1) != 1))