Counter Strike : Global Offensive Source Code
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

448 lines
14 KiB

  1. /* gzjoin -- command to join gzip files into one gzip file
  2. Copyright (C) 2004 Mark Adler, all rights reserved
  3. version 1.0, 11 Dec 2004
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the author be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. Mark Adler madler@alumni.caltech.edu
  18. */
  19. /*
  20. * Change history:
  21. *
  22. * 1.0 11 Dec 2004 - First version
  23. * 1.1 12 Jun 2005 - Changed ssize_t to long for portability
  24. */
  25. /*
  26. gzjoin takes one or more gzip files on the command line and writes out a
  27. single gzip file that will uncompress to the concatenation of the
  28. uncompressed data from the individual gzip files. gzjoin does this without
  29. having to recompress any of the data and without having to calculate a new
  30. crc32 for the concatenated uncompressed data. gzjoin does however have to
  31. decompress all of the input data in order to find the bits in the compressed
  32. data that need to be modified to concatenate the streams.
  33. gzjoin does not do an integrity check on the input gzip files other than
  34. checking the gzip header and decompressing the compressed data. They are
  35. otherwise assumed to be complete and correct.
  36. Each joint between gzip files removes at least 18 bytes of previous trailer
  37. and subsequent header, and inserts an average of about three bytes to the
  38. compressed data in order to connect the streams. The output gzip file
  39. has a minimal ten-byte gzip header with no file name or modification time.
  40. This program was written to illustrate the use of the Z_BLOCK option of
  41. inflate() and the crc32_combine() function. gzjoin will not compile with
  42. versions of zlib earlier than 1.2.3.
  43. */
  44. #include <stdio.h> /* fputs(), fprintf(), fwrite(), putc() */
  45. #include <stdlib.h> /* exit(), malloc(), free() */
  46. #include <fcntl.h> /* open() */
  47. #include <unistd.h> /* close(), read(), lseek() */
  48. #include "zlib.h"
  49. /* crc32(), crc32_combine(), inflateInit2(), inflate(), inflateEnd() */
  50. #define local static
  51. /* exit with an error (return a value to allow use in an expression) */
  52. local int bail(char *why1, char *why2)
  53. {
  54. fprintf(stderr, "gzjoin error: %s%s, output incomplete\n", why1, why2);
  55. exit(1);
  56. return 0;
  57. }
  58. /* -- simple buffered file input with access to the buffer -- */
  59. #define CHUNK 32768 /* must be a power of two and fit in unsigned */
  60. /* bin buffered input file type */
  61. typedef struct {
  62. char *name; /* name of file for error messages */
  63. int fd; /* file descriptor */
  64. unsigned left; /* bytes remaining at next */
  65. unsigned char *next; /* next byte to read */
  66. unsigned char *buf; /* allocated buffer of length CHUNK */
  67. } bin;
  68. /* close a buffered file and free allocated memory */
  69. local void bclose(bin *in)
  70. {
  71. if (in != NULL) {
  72. if (in->fd != -1)
  73. close(in->fd);
  74. if (in->buf != NULL)
  75. free(in->buf);
  76. free(in);
  77. }
  78. }
  79. /* open a buffered file for input, return a pointer to type bin, or NULL on
  80. failure */
  81. local bin *bopen(char *name)
  82. {
  83. bin *in;
  84. in = malloc(sizeof(bin));
  85. if (in == NULL)
  86. return NULL;
  87. in->buf = malloc(CHUNK);
  88. in->fd = open(name, O_RDONLY, 0);
  89. if (in->buf == NULL || in->fd == -1) {
  90. bclose(in);
  91. return NULL;
  92. }
  93. in->left = 0;
  94. in->next = in->buf;
  95. in->name = name;
  96. return in;
  97. }
  98. /* load buffer from file, return -1 on read error, 0 or 1 on success, with
  99. 1 indicating that end-of-file was reached */
  100. local int bload(bin *in)
  101. {
  102. long len;
  103. if (in == NULL)
  104. return -1;
  105. if (in->left != 0)
  106. return 0;
  107. in->next = in->buf;
  108. do {
  109. len = (long)read(in->fd, in->buf + in->left, CHUNK - in->left);
  110. if (len < 0)
  111. return -1;
  112. in->left += (unsigned)len;
  113. } while (len != 0 && in->left < CHUNK);
  114. return len == 0 ? 1 : 0;
  115. }
  116. /* get a byte from the file, bail if end of file */
  117. #define bget(in) (in->left ? 0 : bload(in), \
  118. in->left ? (in->left--, *(in->next)++) : \
  119. bail("unexpected end of file on ", in->name))
  120. /* get a four-byte little-endian unsigned integer from file */
  121. local unsigned long bget4(bin *in)
  122. {
  123. unsigned long val;
  124. val = bget(in);
  125. val += (unsigned long)(bget(in)) << 8;
  126. val += (unsigned long)(bget(in)) << 16;
  127. val += (unsigned long)(bget(in)) << 24;
  128. return val;
  129. }
  130. /* skip bytes in file */
  131. local void bskip(bin *in, unsigned skip)
  132. {
  133. /* check pointer */
  134. if (in == NULL)
  135. return;
  136. /* easy case -- skip bytes in buffer */
  137. if (skip <= in->left) {
  138. in->left -= skip;
  139. in->next += skip;
  140. return;
  141. }
  142. /* skip what's in buffer, discard buffer contents */
  143. skip -= in->left;
  144. in->left = 0;
  145. /* seek past multiples of CHUNK bytes */
  146. if (skip > CHUNK) {
  147. unsigned left;
  148. left = skip & (CHUNK - 1);
  149. if (left == 0) {
  150. /* exact number of chunks: seek all the way minus one byte to check
  151. for end-of-file with a read */
  152. lseek(in->fd, skip - 1, SEEK_CUR);
  153. if (read(in->fd, in->buf, 1) != 1)
  154. bail("unexpected end of file on ", in->name);
  155. return;
  156. }
  157. /* skip the integral chunks, update skip with remainder */
  158. lseek(in->fd, skip - left, SEEK_CUR);
  159. skip = left;
  160. }
  161. /* read more input and skip remainder */
  162. bload(in);
  163. if (skip > in->left)
  164. bail("unexpected end of file on ", in->name);
  165. in->left -= skip;
  166. in->next += skip;
  167. }
  168. /* -- end of buffered input functions -- */
  169. /* skip the gzip header from file in */
  170. local void gzhead(bin *in)
  171. {
  172. int flags;
  173. /* verify gzip magic header and compression method */
  174. if (bget(in) != 0x1f || bget(in) != 0x8b || bget(in) != 8)
  175. bail(in->name, " is not a valid gzip file");
  176. /* get and verify flags */
  177. flags = bget(in);
  178. if ((flags & 0xe0) != 0)
  179. bail("unknown reserved bits set in ", in->name);
  180. /* skip modification time, extra flags, and os */
  181. bskip(in, 6);
  182. /* skip extra field if present */
  183. if (flags & 4) {
  184. unsigned len;
  185. len = bget(in);
  186. len += (unsigned)(bget(in)) << 8;
  187. bskip(in, len);
  188. }
  189. /* skip file name if present */
  190. if (flags & 8)
  191. while (bget(in) != 0)
  192. ;
  193. /* skip comment if present */
  194. if (flags & 16)
  195. while (bget(in) != 0)
  196. ;
  197. /* skip header crc if present */
  198. if (flags & 2)
  199. bskip(in, 2);
  200. }
  201. /* write a four-byte little-endian unsigned integer to out */
  202. local void put4(unsigned long val, FILE *out)
  203. {
  204. putc(val & 0xff, out);
  205. putc((val >> 8) & 0xff, out);
  206. putc((val >> 16) & 0xff, out);
  207. putc((val >> 24) & 0xff, out);
  208. }
  209. /* Load up zlib stream from buffered input, bail if end of file */
  210. local void zpull(z_streamp strm, bin *in)
  211. {
  212. if (in->left == 0)
  213. bload(in);
  214. if (in->left == 0)
  215. bail("unexpected end of file on ", in->name);
  216. strm->avail_in = in->left;
  217. strm->next_in = in->next;
  218. }
  219. /* Write header for gzip file to out and initialize trailer. */
  220. local void gzinit(unsigned long *crc, unsigned long *tot, FILE *out)
  221. {
  222. fwrite("\x1f\x8b\x08\0\0\0\0\0\0\xff", 1, 10, out);
  223. *crc = crc32(0L, Z_NULL, 0);
  224. *tot = 0;
  225. }
  226. /* Copy the compressed data from name, zeroing the last block bit of the last
  227. block if clr is true, and adding empty blocks as needed to get to a byte
  228. boundary. If clr is false, then the last block becomes the last block of
  229. the output, and the gzip trailer is written. crc and tot maintains the
  230. crc and length (modulo 2^32) of the output for the trailer. The resulting
  231. gzip file is written to out. gzinit() must be called before the first call
  232. of gzcopy() to write the gzip header and to initialize crc and tot. */
  233. local void gzcopy(char *name, int clr, unsigned long *crc, unsigned long *tot,
  234. FILE *out)
  235. {
  236. int ret; /* return value from zlib functions */
  237. int pos; /* where the "last block" bit is in byte */
  238. int last; /* true if processing the last block */
  239. bin *in; /* buffered input file */
  240. unsigned char *start; /* start of compressed data in buffer */
  241. unsigned char *junk; /* buffer for uncompressed data -- discarded */
  242. z_off_t len; /* length of uncompressed data (support > 4 GB) */
  243. z_stream strm; /* zlib inflate stream */
  244. /* open gzip file and skip header */
  245. in = bopen(name);
  246. if (in == NULL)
  247. bail("could not open ", name);
  248. gzhead(in);
  249. /* allocate buffer for uncompressed data and initialize raw inflate
  250. stream */
  251. junk = malloc(CHUNK);
  252. strm.zalloc = Z_NULL;
  253. strm.zfree = Z_NULL;
  254. strm.opaque = Z_NULL;
  255. strm.avail_in = 0;
  256. strm.next_in = Z_NULL;
  257. ret = inflateInit2(&strm, -15);
  258. if (junk == NULL || ret != Z_OK)
  259. bail("out of memory", "");
  260. /* inflate and copy compressed data, clear last-block bit if requested */
  261. len = 0;
  262. zpull(&strm, in);
  263. start = strm.next_in;
  264. last = start[0] & 1;
  265. if (last && clr)
  266. start[0] &= ~1;
  267. strm.avail_out = 0;
  268. for (;;) {
  269. /* if input used and output done, write used input and get more */
  270. if (strm.avail_in == 0 && strm.avail_out != 0) {
  271. fwrite(start, 1, strm.next_in - start, out);
  272. start = in->buf;
  273. in->left = 0;
  274. zpull(&strm, in);
  275. }
  276. /* decompress -- return early when end-of-block reached */
  277. strm.avail_out = CHUNK;
  278. strm.next_out = junk;
  279. ret = inflate(&strm, Z_BLOCK);
  280. switch (ret) {
  281. case Z_MEM_ERROR:
  282. bail("out of memory", "");
  283. case Z_DATA_ERROR:
  284. bail("invalid compressed data in ", in->name);
  285. }
  286. /* update length of uncompressed data */
  287. len += CHUNK - strm.avail_out;
  288. /* check for block boundary (only get this when block copied out) */
  289. if (strm.data_type & 128) {
  290. /* if that was the last block, then done */
  291. if (last)
  292. break;
  293. /* number of unused bits in last byte */
  294. pos = strm.data_type & 7;
  295. /* find the next last-block bit */
  296. if (pos != 0) {
  297. /* next last-block bit is in last used byte */
  298. pos = 0x100 >> pos;
  299. last = strm.next_in[-1] & pos;
  300. if (last && clr)
  301. strm.next_in[-1] &= ~pos;
  302. }
  303. else {
  304. /* next last-block bit is in next unused byte */
  305. if (strm.avail_in == 0) {
  306. /* don't have that byte yet -- get it */
  307. fwrite(start, 1, strm.next_in - start, out);
  308. start = in->buf;
  309. in->left = 0;
  310. zpull(&strm, in);
  311. }
  312. last = strm.next_in[0] & 1;
  313. if (last && clr)
  314. strm.next_in[0] &= ~1;
  315. }
  316. }
  317. }
  318. /* update buffer with unused input */
  319. in->left = strm.avail_in;
  320. in->next = strm.next_in;
  321. /* copy used input, write empty blocks to get to byte boundary */
  322. pos = strm.data_type & 7;
  323. fwrite(start, 1, in->next - start - 1, out);
  324. last = in->next[-1];
  325. if (pos == 0 || !clr)
  326. /* already at byte boundary, or last file: write last byte */
  327. putc(last, out);
  328. else {
  329. /* append empty blocks to last byte */
  330. last &= ((0x100 >> pos) - 1); /* assure unused bits are zero */
  331. if (pos & 1) {
  332. /* odd -- append an empty stored block */
  333. putc(last, out);
  334. if (pos == 1)
  335. putc(0, out); /* two more bits in block header */
  336. fwrite("\0\0\xff\xff", 1, 4, out);
  337. }
  338. else {
  339. /* even -- append 1, 2, or 3 empty fixed blocks */
  340. switch (pos) {
  341. case 6:
  342. putc(last | 8, out);
  343. last = 0;
  344. case 4:
  345. putc(last | 0x20, out);
  346. last = 0;
  347. case 2:
  348. putc(last | 0x80, out);
  349. putc(0, out);
  350. }
  351. }
  352. }
  353. /* update crc and tot */
  354. *crc = crc32_combine(*crc, bget4(in), len);
  355. *tot += (unsigned long)len;
  356. /* clean up */
  357. inflateEnd(&strm);
  358. free(junk);
  359. bclose(in);
  360. /* write trailer if this is the last gzip file */
  361. if (!clr) {
  362. put4(*crc, out);
  363. put4(*tot, out);
  364. }
  365. }
  366. /* join the gzip files on the command line, write result to stdout */
  367. int main(int argc, char **argv)
  368. {
  369. unsigned long crc, tot; /* running crc and total uncompressed length */
  370. /* skip command name */
  371. argc--;
  372. argv++;
  373. /* show usage if no arguments */
  374. if (argc == 0) {
  375. fputs("gzjoin usage: gzjoin f1.gz [f2.gz [f3.gz ...]] > fjoin.gz\n",
  376. stderr);
  377. return 0;
  378. }
  379. /* join gzip files on command line and write to stdout */
  380. gzinit(&crc, &tot, stdout);
  381. while (argc--)
  382. gzcopy(*argv++, argc, &crc, &tot, stdout);
  383. /* done */
  384. return 0;
  385. }