2  * Copyright (c) 2005, 2006 Rene Scharfe
 
   7 #include "run-command.h"
 
   9 #define RECORDSIZE      (512)
 
  10 #define BLOCKSIZE       (RECORDSIZE * 20)
 
  12 static char block[BLOCKSIZE];
 
  13 static unsigned long offset;
 
  15 static int tar_umask = 002;
 
  17 static int write_tar_filter_archive(const struct archiver *ar,
 
  18                                     struct archiver_args *args);
 
  20 /* writes out the whole block, but only if it is full */
 
  21 static void write_if_needed(void)
 
  23         if (offset == BLOCKSIZE) {
 
  24                 write_or_die(1, block, BLOCKSIZE);
 
  30  * queues up writes, so that all our write(2) calls write exactly one
 
  31  * full block; pads writes to RECORDSIZE
 
  33 static void write_blocked(const void *data, unsigned long size)
 
  35         const char *buf = data;
 
  39                 unsigned long chunk = BLOCKSIZE - offset;
 
  42                 memcpy(block + offset, buf, chunk);
 
  48         while (size >= BLOCKSIZE) {
 
  49                 write_or_die(1, buf, BLOCKSIZE);
 
  54                 memcpy(block + offset, buf, size);
 
  57         tail = offset % RECORDSIZE;
 
  59                 memset(block + offset, 0, RECORDSIZE - tail);
 
  60                 offset += RECORDSIZE - tail;
 
  66  * The end of tar archives is marked by 2*512 nul bytes and after that
 
  67  * follows the rest of the block (if any).
 
  69 static void write_trailer(void)
 
  71         int tail = BLOCKSIZE - offset;
 
  72         memset(block + offset, 0, tail);
 
  73         write_or_die(1, block, BLOCKSIZE);
 
  74         if (tail < 2 * RECORDSIZE) {
 
  75                 memset(block, 0, offset);
 
  76                 write_or_die(1, block, BLOCKSIZE);
 
  81  * pax extended header records have the format "%u %s=%s\n".  %u contains
 
  82  * the size of the whole string (including the %u), the first %s is the
 
  83  * keyword, the second one is the value.  This function constructs such a
 
  84  * string and appends it to a struct strbuf.
 
  86 static void strbuf_append_ext_header(struct strbuf *sb, const char *keyword,
 
  87                                      const char *value, unsigned int valuelen)
 
  92         len = 1 + 1 + strlen(keyword) + 1 + valuelen + 1;
 
  93         for (tmp = len; tmp > 9; tmp /= 10)
 
  97         strbuf_addf(sb, "%u %s=", len, keyword);
 
  98         strbuf_add(sb, value, valuelen);
 
  99         strbuf_addch(sb, '\n');
 
 102 static unsigned int ustar_header_chksum(const struct ustar_header *header)
 
 104         char *p = (char *)header;
 
 105         unsigned int chksum = 0;
 
 106         while (p < header->chksum)
 
 108         chksum += sizeof(header->chksum) * ' ';
 
 109         p += sizeof(header->chksum);
 
 110         while (p < (char *)header + sizeof(struct ustar_header))
 
 115 static size_t get_path_prefix(const char *path, size_t pathlen, size_t maxlen)
 
 122         } while (i > 0 && path[i] != '/');
 
 126 static int write_tar_entry(struct archiver_args *args,
 
 127                 const unsigned char *sha1, const char *path, size_t pathlen,
 
 128                 unsigned int mode, void *buffer, unsigned long size)
 
 130         struct ustar_header header;
 
 131         struct strbuf ext_header = STRBUF_INIT;
 
 134         memset(&header, 0, sizeof(header));
 
 137                 *header.typeflag = TYPEFLAG_GLOBAL_HEADER;
 
 139                 strcpy(header.name, "pax_global_header");
 
 141                 *header.typeflag = TYPEFLAG_EXT_HEADER;
 
 143                 sprintf(header.name, "%s.paxheader", sha1_to_hex(sha1));
 
 145                 if (S_ISDIR(mode) || S_ISGITLINK(mode)) {
 
 146                         *header.typeflag = TYPEFLAG_DIR;
 
 147                         mode = (mode | 0777) & ~tar_umask;
 
 148                 } else if (S_ISLNK(mode)) {
 
 149                         *header.typeflag = TYPEFLAG_LNK;
 
 151                 } else if (S_ISREG(mode)) {
 
 152                         *header.typeflag = TYPEFLAG_REG;
 
 153                         mode = (mode | ((mode & 0100) ? 0777 : 0666)) & ~tar_umask;
 
 155                         return error("unsupported file mode: 0%o (SHA1: %s)",
 
 156                                         mode, sha1_to_hex(sha1));
 
 158                 if (pathlen > sizeof(header.name)) {
 
 159                         size_t plen = get_path_prefix(path, pathlen,
 
 160                                         sizeof(header.prefix));
 
 161                         size_t rest = pathlen - plen - 1;
 
 162                         if (plen > 0 && rest <= sizeof(header.name)) {
 
 163                                 memcpy(header.prefix, path, plen);
 
 164                                 memcpy(header.name, path + plen + 1, rest);
 
 166                                 sprintf(header.name, "%s.data",
 
 168                                 strbuf_append_ext_header(&ext_header, "path",
 
 172                         memcpy(header.name, path, pathlen);
 
 175         if (S_ISLNK(mode) && buffer) {
 
 176                 if (size > sizeof(header.linkname)) {
 
 177                         sprintf(header.linkname, "see %s.paxheader",
 
 179                         strbuf_append_ext_header(&ext_header, "linkpath",
 
 182                         memcpy(header.linkname, buffer, size);
 
 185         sprintf(header.mode, "%07o", mode & 07777);
 
 186         sprintf(header.size, "%011lo", S_ISREG(mode) ? size : 0);
 
 187         sprintf(header.mtime, "%011lo", (unsigned long) args->time);
 
 189         sprintf(header.uid, "%07o", 0);
 
 190         sprintf(header.gid, "%07o", 0);
 
 191         strlcpy(header.uname, "root", sizeof(header.uname));
 
 192         strlcpy(header.gname, "root", sizeof(header.gname));
 
 193         sprintf(header.devmajor, "%07o", 0);
 
 194         sprintf(header.devminor, "%07o", 0);
 
 196         memcpy(header.magic, "ustar", 6);
 
 197         memcpy(header.version, "00", 2);
 
 199         sprintf(header.chksum, "%07o", ustar_header_chksum(&header));
 
 201         if (ext_header.len > 0) {
 
 202                 err = write_tar_entry(args, sha1, NULL, 0, 0, ext_header.buf,
 
 207         strbuf_release(&ext_header);
 
 208         write_blocked(&header, sizeof(header));
 
 209         if (S_ISREG(mode) && buffer && size > 0)
 
 210                 write_blocked(buffer, size);
 
 214 static int write_global_extended_header(struct archiver_args *args)
 
 216         const unsigned char *sha1 = args->commit_sha1;
 
 217         struct strbuf ext_header = STRBUF_INIT;
 
 220         strbuf_append_ext_header(&ext_header, "comment", sha1_to_hex(sha1), 40);
 
 221         err = write_tar_entry(args, NULL, NULL, 0, 0, ext_header.buf,
 
 223         strbuf_release(&ext_header);
 
 227 static struct archiver **tar_filters;
 
 228 static int nr_tar_filters;
 
 229 static int alloc_tar_filters;
 
 231 static struct archiver *find_tar_filter(const char *name, int len)
 
 234         for (i = 0; i < nr_tar_filters; i++) {
 
 235                 struct archiver *ar = tar_filters[i];
 
 236                 if (!strncmp(ar->name, name, len) && !ar->name[len])
 
 242 static int tar_filter_config(const char *var, const char *value, void *data)
 
 250         if (prefixcmp(var, "tar."))
 
 252         dot = strrchr(var, '.');
 
 257         namelen = dot - name;
 
 260         ar = find_tar_filter(name, namelen);
 
 262                 ar = xcalloc(1, sizeof(*ar));
 
 263                 ar->name = xmemdupz(name, namelen);
 
 264                 ar->write_archive = write_tar_filter_archive;
 
 265                 ar->flags = ARCHIVER_WANT_COMPRESSION_LEVELS;
 
 266                 ALLOC_GROW(tar_filters, nr_tar_filters + 1, alloc_tar_filters);
 
 267                 tar_filters[nr_tar_filters++] = ar;
 
 270         if (!strcmp(type, "command")) {
 
 272                         return config_error_nonbool(var);
 
 274                 ar->data = xstrdup(value);
 
 281 static int git_tar_config(const char *var, const char *value, void *cb)
 
 283         if (!strcmp(var, "tar.umask")) {
 
 284                 if (value && !strcmp(value, "user")) {
 
 285                         tar_umask = umask(0);
 
 288                         tar_umask = git_config_int(var, value);
 
 293         return tar_filter_config(var, value, cb);
 
 296 static int write_tar_archive(const struct archiver *ar,
 
 297                              struct archiver_args *args)
 
 301         if (args->commit_sha1)
 
 302                 err = write_global_extended_header(args);
 
 304                 err = write_archive_entries(args, write_tar_entry);
 
 310 static int write_tar_filter_archive(const struct archiver *ar,
 
 311                                     struct archiver_args *args)
 
 313         struct strbuf cmd = STRBUF_INIT;
 
 314         struct child_process filter;
 
 319                 die("BUG: tar-filter archiver called with no filter defined");
 
 321         strbuf_addstr(&cmd, ar->data);
 
 322         if (args->compression_level >= 0)
 
 323                 strbuf_addf(&cmd, " -%d", args->compression_level);
 
 325         memset(&filter, 0, sizeof(filter));
 
 329         filter.use_shell = 1;
 
 332         if (start_command(&filter) < 0)
 
 333                 die_errno("unable to start '%s' filter", argv[0]);
 
 335         if (dup2(filter.in, 1) < 0)
 
 336                 die_errno("unable to redirect descriptor");
 
 339         r = write_tar_archive(ar, args);
 
 342         if (finish_command(&filter) != 0)
 
 343                 die("'%s' filter reported error", argv[0]);
 
 345         strbuf_release(&cmd);
 
 349 static struct archiver tar_archiver = {
 
 355 void init_tar_archiver(void)
 
 358         register_archiver(&tar_archiver);
 
 360         git_config(git_tar_config, NULL);
 
 361         for (i = 0; i < nr_tar_filters; i++) {
 
 362                 /* omit any filters that never had a command configured */
 
 363                 if (tar_filters[i]->data)
 
 364                         register_archiver(tar_filters[i]);