2 /*-----------------------------------------------------------*/
3 /*--- A block-sorting, lossless compressor bzip2.c ---*/
4 /*-----------------------------------------------------------*/
7 This file is a part of bzip2 and/or libbzip2, a program and
8 library for lossless, block-sorting data compression.
10 Copyright (C) 1996-2005 Julian R Seward. All rights reserved.
12 Redistribution and use in source and binary forms, with or without
13 modification, are permitted provided that the following conditions
16 1. Redistributions of source code must retain the above copyright
17 notice, this list of conditions and the following disclaimer.
19 2. The origin of this software must not be misrepresented; you must
20 not claim that you wrote the original software. If you use this
21 software in a product, an acknowledgment in the product
22 documentation would be appreciated but is not required.
24 3. Altered source versions must be plainly marked as such, and must
25 not be misrepresented as being the original software.
27 4. The name of the author may not be used to endorse or promote
28 products derived from this software without specific prior written
31 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
32 OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
33 WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
34 ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
35 DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36 DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
37 GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
38 INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
39 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
40 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
41 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
43 Julian Seward, Cambridge, UK.
45 bzip2/libbzip2 version 1.0 of 21 March 2000
47 This program is based on (at least) the work of:
57 For more information on these sources, see the manual.
61 /*----------------------------------------------------*/
63 /*----------------------------------------------------*/
67 This program and library (attempts to) compress data by
68 performing several non-trivial transformations on it.
69 Unless you are 100% familiar with *all* the algorithms
70 contained herein, and with the consequences of modifying them,
71 you should NOT meddle with the compression or decompression
72 machinery. Incorrect changes can and very likely *will*
73 lead to disasterous loss of data.
76 I TAKE NO RESPONSIBILITY FOR ANY LOSS OF DATA ARISING FROM THE
77 USE OF THIS PROGRAM, HOWSOEVER CAUSED.
79 Every compression of a file implies an assumption that the
80 compressed file can be decompressed to reproduce the original.
81 Great efforts in design, coding and testing have been made to
82 ensure that this program works correctly. However, the
83 complexity of the algorithms, and, in particular, the presence
84 of various special cases in the code which occur with very low
85 but non-zero probability make it impossible to rule out the
86 possibility of bugs remaining in the program. DO NOT COMPRESS
87 ANY DATA WITH THIS PROGRAM AND/OR LIBRARY UNLESS YOU ARE PREPARED
88 TO ACCEPT THE POSSIBILITY, HOWEVER SMALL, THAT THE DATA WILL
91 That is not to say this program is inherently unreliable.
92 Indeed, I very much hope the opposite is true. bzip2/libbzip2
93 has been carefully constructed and extensively tested.
96 To the best of my knowledge, bzip2/libbzip2 does not use any
97 patented algorithms. However, I do not have the resources
98 available to carry out a full patent search. Therefore I cannot
99 give any guarantee of the above statement.
104 /*----------------------------------------------------*/
105 /*--- and now for something much more pleasant :-) ---*/
106 /*----------------------------------------------------*/
108 /*---------------------------------------------*/
110 Place a 1 beside your platform, and 0 elsewhere.
115 Also works on 64-bit Unix boxes.
121 Win32, as seen by Jacob Navia's excellent
122 port of (Chris Fraser & David Hanson)'s excellent
123 lcc compiler. Or with MS Visual C.
124 This is selected automatically if compiled by a compiler which
125 defines _WIN32, not including the Cygwin GCC.
127 #define BZ_LCCWIN32 0
129 #if defined(_WIN32) && !defined(__CYGWIN__)
131 #define BZ_LCCWIN32 1
137 /*---------------------------------------------*/
139 Some stuff for all platforms.
151 #define ERROR_IF_EOF(i) { if ((i) == EOF) ioError(); }
152 #define ERROR_IF_NOT_ZERO(i) { if ((i) != 0) ioError(); }
153 #define ERROR_IF_MINUS_ONE(i) { if ((i) == (-1)) ioError(); }
156 /*---------------------------------------------*/
158 Platform-specific stuff.
163 # include <sys/types.h>
166 # include <sys/stat.h>
167 # include <sys/times.h>
169 # define PATH_SEP '/'
170 # define MY_LSTAT lstat
171 # define MY_STAT stat
172 # define MY_S_ISREG S_ISREG
173 # define MY_S_ISDIR S_ISDIR
175 # define APPEND_FILESPEC(root, name) \
176 root=snocString((root), (name))
178 # define APPEND_FLAG(root, name) \
179 root=snocString((root), (name))
181 # define SET_BINARY_MODE(fd) /**/
184 # define NORETURN __attribute__ ((noreturn))
186 # define NORETURN /**/
194 # define MY_LSTAT stat
195 # define MY_STAT stat
196 # undef SET_BINARY_MODE
197 # define SET_BINARY_MODE(fd) \
199 int retVal = setmode ( fileno ( fd ), \
201 ERROR_IF_MINUS_ONE ( retVal ); \
208 # undef SET_BINARY_MODE
209 # define SET_BINARY_MODE(fd) \
211 int retVal = setmode ( fileno ( fd ), \
213 ERROR_IF_MINUS_ONE ( retVal ); \
223 # include <sys\stat.h>
225 # define NORETURN /**/
226 # define PATH_SEP '\\'
227 # define MY_LSTAT _stat
228 # define MY_STAT _stat
229 # define MY_S_ISREG(x) ((x) & _S_IFREG)
230 # define MY_S_ISDIR(x) ((x) & _S_IFDIR)
232 # define APPEND_FLAG(root, name) \
233 root=snocString((root), (name))
235 # define APPEND_FILESPEC(root, name) \
236 root = snocString ((root), (name))
238 # define SET_BINARY_MODE(fd) \
240 int retVal = setmode ( fileno ( fd ), \
242 ERROR_IF_MINUS_ONE ( retVal ); \
245 #endif /* BZ_LCCWIN32 */
248 /*---------------------------------------------*/
250 Some more stuff for all platforms :-)
254 typedef unsigned char Bool
;
255 typedef unsigned char UChar
;
257 typedef unsigned int UInt32
;
259 typedef unsigned short UInt16
;
261 #define True ((Bool)1)
262 #define False ((Bool)0)
265 IntNative is your platform's `native' int size.
266 Only here to avoid probs with 64-bit platforms.
268 typedef int IntNative
;
271 /*---------------------------------------------------*/
272 /*--- Misc (file handling) data decls ---*/
273 /*---------------------------------------------------*/
276 Bool keepInputFiles
, smallMode
, deleteOutputOnInterrupt
;
277 Bool forceOverwrite
, testFailsExist
, unzFailsExist
, noisy
;
278 Int32 numFileNames
, numFilesProcessed
, blockSize100k
;
281 /*-- source modes; F==file, I==stdin, O==stdout --*/
286 /*-- operation modes --*/
294 #define FILE_NAME_LEN 1034
296 Int32 longestFileName
;
297 Char inName
[FILE_NAME_LEN
];
298 Char outName
[FILE_NAME_LEN
];
299 Char tmpName
[FILE_NAME_LEN
];
301 Char progNameReally
[FILE_NAME_LEN
];
302 FILE *outputHandleJustInCase
;
305 static void panic ( Char
* ) NORETURN
;
306 static void ioError ( void ) NORETURN
;
307 static void outOfMemory ( void ) NORETURN
;
308 static void configError ( void ) NORETURN
;
309 static void crcError ( void ) NORETURN
;
310 static void cleanUpAndFail ( Int32
) NORETURN
;
311 static void compressedStreamEOF ( void ) NORETURN
;
313 static void copyFileName ( Char
*, Char
* );
314 static void* myMalloc ( Int32
);
318 /*---------------------------------------------------*/
319 /*--- An implementation of 64-bit ints. Sigh. ---*/
320 /*--- Roll on widespread deployment of ANSI C9X ! ---*/
321 /*---------------------------------------------------*/
324 struct { UChar b
[8]; }
329 void uInt64_from_UInt32s ( UInt64
* n
, UInt32 lo32
, UInt32 hi32
)
331 n
->b
[7] = (UChar
)((hi32
>> 24) & 0xFF);
332 n
->b
[6] = (UChar
)((hi32
>> 16) & 0xFF);
333 n
->b
[5] = (UChar
)((hi32
>> 8) & 0xFF);
334 n
->b
[4] = (UChar
) (hi32
& 0xFF);
335 n
->b
[3] = (UChar
)((lo32
>> 24) & 0xFF);
336 n
->b
[2] = (UChar
)((lo32
>> 16) & 0xFF);
337 n
->b
[1] = (UChar
)((lo32
>> 8) & 0xFF);
338 n
->b
[0] = (UChar
) (lo32
& 0xFF);
343 double uInt64_to_double ( UInt64
* n
)
348 for (i
= 0; i
< 8; i
++) {
349 sum
+= base
* (double)(n
->b
[i
]);
357 Bool
uInt64_isZero ( UInt64
* n
)
360 for (i
= 0; i
< 8; i
++)
361 if (n
->b
[i
] != 0) return 0;
366 /* Divide *n by 10, and return the remainder. */
368 Int32
uInt64_qrm10 ( UInt64
* n
)
373 for (i
= 7; i
>= 0; i
--) {
374 tmp
= rem
* 256 + n
->b
[i
];
382 /* ... and the Whole Entire Point of all this UInt64 stuff is
383 so that we can supply the following function.
386 void uInt64_toAscii ( char* outbuf
, UInt64
* n
)
393 q
= uInt64_qrm10 ( &n_copy
);
396 } while (!uInt64_isZero(&n_copy
));
398 for (i
= 0; i
< nBuf
; i
++)
399 outbuf
[i
] = buf
[nBuf
-i
-1];
403 /*---------------------------------------------------*/
404 /*--- Processing of complete files and streams ---*/
405 /*---------------------------------------------------*/
407 /*---------------------------------------------*/
409 Bool
myfeof ( FILE* f
)
411 Int32 c
= fgetc ( f
);
412 if (c
== EOF
) return True
;
418 /*---------------------------------------------*/
420 void compressStream ( FILE *stream
, FILE *zStream
)
425 UInt32 nbytes_in_lo32
, nbytes_in_hi32
;
426 UInt32 nbytes_out_lo32
, nbytes_out_hi32
;
427 Int32 bzerr
, bzerr_dummy
, ret
;
429 SET_BINARY_MODE(stream
);
430 SET_BINARY_MODE(zStream
);
432 if (ferror(stream
)) goto errhandler_io
;
433 if (ferror(zStream
)) goto errhandler_io
;
435 bzf
= BZ2_bzWriteOpen ( &bzerr
, zStream
,
436 blockSize100k
, verbosity
, workFactor
);
437 if (bzerr
!= BZ_OK
) goto errhandler
;
439 if (verbosity
>= 2) fprintf ( stderr
, "\n" );
443 if (myfeof(stream
)) break;
444 nIbuf
= fread ( ibuf
, sizeof(UChar
), 5000, stream
);
445 if (ferror(stream
)) goto errhandler_io
;
446 if (nIbuf
> 0) BZ2_bzWrite ( &bzerr
, bzf
, (void*)ibuf
, nIbuf
);
447 if (bzerr
!= BZ_OK
) goto errhandler
;
451 BZ2_bzWriteClose64 ( &bzerr
, bzf
, 0,
452 &nbytes_in_lo32
, &nbytes_in_hi32
,
453 &nbytes_out_lo32
, &nbytes_out_hi32
);
454 if (bzerr
!= BZ_OK
) goto errhandler
;
456 if (ferror(zStream
)) goto errhandler_io
;
457 ret
= fflush ( zStream
);
458 if (ret
== EOF
) goto errhandler_io
;
459 if (zStream
!= stdout
) {
460 ret
= fclose ( zStream
);
461 outputHandleJustInCase
= NULL
;
462 if (ret
== EOF
) goto errhandler_io
;
464 outputHandleJustInCase
= NULL
;
465 if (ferror(stream
)) goto errhandler_io
;
466 ret
= fclose ( stream
);
467 if (ret
== EOF
) goto errhandler_io
;
469 if (verbosity
>= 1) {
470 if (nbytes_in_lo32
== 0 && nbytes_in_hi32
== 0) {
471 fprintf ( stderr
, " no data compressed.\n");
473 Char buf_nin
[32], buf_nout
[32];
474 UInt64 nbytes_in
, nbytes_out
;
475 double nbytes_in_d
, nbytes_out_d
;
476 uInt64_from_UInt32s ( &nbytes_in
,
477 nbytes_in_lo32
, nbytes_in_hi32
);
478 uInt64_from_UInt32s ( &nbytes_out
,
479 nbytes_out_lo32
, nbytes_out_hi32
);
480 nbytes_in_d
= uInt64_to_double ( &nbytes_in
);
481 nbytes_out_d
= uInt64_to_double ( &nbytes_out
);
482 uInt64_toAscii ( buf_nin
, &nbytes_in
);
483 uInt64_toAscii ( buf_nout
, &nbytes_out
);
484 fprintf ( stderr
, "%6.3f:1, %6.3f bits/byte, "
485 "%5.2f%% saved, %s in, %s out.\n",
486 nbytes_in_d
/ nbytes_out_d
,
487 (8.0 * nbytes_out_d
) / nbytes_in_d
,
488 100.0 * (1.0 - nbytes_out_d
/ nbytes_in_d
),
498 BZ2_bzWriteClose64 ( &bzerr_dummy
, bzf
, 1,
499 &nbytes_in_lo32
, &nbytes_in_hi32
,
500 &nbytes_out_lo32
, &nbytes_out_hi32
);
502 case BZ_CONFIG_ERROR
:
503 configError(); break;
505 outOfMemory (); break;
510 panic ( "compress:unexpected error" );
513 panic ( "compress:end" );
519 /*---------------------------------------------*/
521 Bool
uncompressStream ( FILE *zStream
, FILE *stream
)
524 Int32 bzerr
, bzerr_dummy
, ret
, nread
, streamNo
, i
;
526 UChar unused
[BZ_MAX_UNUSED
];
534 SET_BINARY_MODE(stream
);
535 SET_BINARY_MODE(zStream
);
537 if (ferror(stream
)) goto errhandler_io
;
538 if (ferror(zStream
)) goto errhandler_io
;
542 bzf
= BZ2_bzReadOpen (
543 &bzerr
, zStream
, verbosity
,
544 (int)smallMode
, unused
, nUnused
546 if (bzf
== NULL
|| bzerr
!= BZ_OK
) goto errhandler
;
549 while (bzerr
== BZ_OK
) {
550 nread
= BZ2_bzRead ( &bzerr
, bzf
, obuf
, 5000 );
551 if (bzerr
== BZ_DATA_ERROR_MAGIC
) goto trycat
;
552 if ((bzerr
== BZ_OK
|| bzerr
== BZ_STREAM_END
) && nread
> 0)
553 fwrite ( obuf
, sizeof(UChar
), nread
, stream
);
554 if (ferror(stream
)) goto errhandler_io
;
556 if (bzerr
!= BZ_STREAM_END
) goto errhandler
;
558 BZ2_bzReadGetUnused ( &bzerr
, bzf
, &unusedTmpV
, &nUnused
);
559 if (bzerr
!= BZ_OK
) panic ( "decompress:bzReadGetUnused" );
561 unusedTmp
= (UChar
*)unusedTmpV
;
562 for (i
= 0; i
< nUnused
; i
++) unused
[i
] = unusedTmp
[i
];
564 BZ2_bzReadClose ( &bzerr
, bzf
);
565 if (bzerr
!= BZ_OK
) panic ( "decompress:bzReadGetUnused" );
567 if (nUnused
== 0 && myfeof(zStream
)) break;
571 if (ferror(zStream
)) goto errhandler_io
;
572 ret
= fclose ( zStream
);
573 if (ret
== EOF
) goto errhandler_io
;
575 if (ferror(stream
)) goto errhandler_io
;
576 ret
= fflush ( stream
);
577 if (ret
!= 0) goto errhandler_io
;
578 if (stream
!= stdout
) {
579 ret
= fclose ( stream
);
580 outputHandleJustInCase
= NULL
;
581 if (ret
== EOF
) goto errhandler_io
;
583 outputHandleJustInCase
= NULL
;
584 if (verbosity
>= 2) fprintf ( stderr
, "\n " );
588 if (forceOverwrite
) {
591 if (myfeof(zStream
)) break;
592 nread
= fread ( obuf
, sizeof(UChar
), 5000, zStream
);
593 if (ferror(zStream
)) goto errhandler_io
;
594 if (nread
> 0) fwrite ( obuf
, sizeof(UChar
), nread
, stream
);
595 if (ferror(stream
)) goto errhandler_io
;
601 BZ2_bzReadClose ( &bzerr_dummy
, bzf
);
603 case BZ_CONFIG_ERROR
:
604 configError(); break;
612 case BZ_UNEXPECTED_EOF
:
613 compressedStreamEOF();
614 case BZ_DATA_ERROR_MAGIC
:
615 if (zStream
!= stdin
) fclose(zStream
);
616 if (stream
!= stdout
) fclose(stream
);
622 "\n%s: %s: trailing garbage after EOF ignored\n",
627 panic ( "decompress:unexpected error" );
630 panic ( "decompress:end" );
631 return True
; /*notreached*/
635 /*---------------------------------------------*/
637 Bool
testStream ( FILE *zStream
)
640 Int32 bzerr
, bzerr_dummy
, ret
, nread
, streamNo
, i
;
642 UChar unused
[BZ_MAX_UNUSED
];
650 SET_BINARY_MODE(zStream
);
651 if (ferror(zStream
)) goto errhandler_io
;
655 bzf
= BZ2_bzReadOpen (
656 &bzerr
, zStream
, verbosity
,
657 (int)smallMode
, unused
, nUnused
659 if (bzf
== NULL
|| bzerr
!= BZ_OK
) goto errhandler
;
662 while (bzerr
== BZ_OK
) {
663 nread
= BZ2_bzRead ( &bzerr
, bzf
, obuf
, 5000 );
664 if (bzerr
== BZ_DATA_ERROR_MAGIC
) goto errhandler
;
666 if (bzerr
!= BZ_STREAM_END
) goto errhandler
;
668 BZ2_bzReadGetUnused ( &bzerr
, bzf
, &unusedTmpV
, &nUnused
);
669 if (bzerr
!= BZ_OK
) panic ( "test:bzReadGetUnused" );
671 unusedTmp
= (UChar
*)unusedTmpV
;
672 for (i
= 0; i
< nUnused
; i
++) unused
[i
] = unusedTmp
[i
];
674 BZ2_bzReadClose ( &bzerr
, bzf
);
675 if (bzerr
!= BZ_OK
) panic ( "test:bzReadGetUnused" );
676 if (nUnused
== 0 && myfeof(zStream
)) break;
680 if (ferror(zStream
)) goto errhandler_io
;
681 ret
= fclose ( zStream
);
682 if (ret
== EOF
) goto errhandler_io
;
684 if (verbosity
>= 2) fprintf ( stderr
, "\n " );
688 BZ2_bzReadClose ( &bzerr_dummy
, bzf
);
690 fprintf ( stderr
, "%s: %s: ", progName
, inName
);
692 case BZ_CONFIG_ERROR
:
693 configError(); break;
699 "data integrity (CRC) error in data\n" );
703 case BZ_UNEXPECTED_EOF
:
705 "file ends unexpectedly\n" );
707 case BZ_DATA_ERROR_MAGIC
:
708 if (zStream
!= stdin
) fclose(zStream
);
711 "bad magic number (file not created by bzip2)\n" );
716 "trailing garbage after EOF ignored\n" );
720 panic ( "test:unexpected error" );
723 panic ( "test:end" );
724 return True
; /*notreached*/
728 /*---------------------------------------------------*/
729 /*--- Error [non-] handling grunge ---*/
730 /*---------------------------------------------------*/
732 /*---------------------------------------------*/
734 void setExit ( Int32 v
)
736 if (v
> exitValue
) exitValue
= v
;
740 /*---------------------------------------------*/
742 void cadvise ( void )
747 "\nIt is possible that the compressed file(s) have become corrupted.\n"
748 "You can use the -tvv option to test integrity of such files.\n\n"
749 "You can use the `bzip2recover' program to attempt to recover\n"
750 "data from undamaged sections of corrupted files.\n\n"
755 /*---------------------------------------------*/
757 void showFileNames ( void )
762 "\tInput file = %s, output file = %s\n",
768 /*---------------------------------------------*/
770 void cleanUpAndFail ( Int32 ec
)
773 struct MY_STAT statBuf
;
775 if ( srcMode
== SM_F2F
777 && deleteOutputOnInterrupt
) {
779 /* Check whether input file still exists. Delete output file
780 only if input exists to avoid loss of data. Joerg Prante, 5
781 January 2002. (JRS 06-Jan-2002: other changes in 1.0.2 mean
782 this is less likely to happen. But to be ultra-paranoid, we
783 do the check anyway.) */
784 retVal
= MY_STAT ( inName
, &statBuf
);
788 "%s: Deleting output file %s, if it exists.\n",
790 if (outputHandleJustInCase
!= NULL
)
791 fclose ( outputHandleJustInCase
);
792 retVal
= remove ( outName
);
795 "%s: WARNING: deletion of output file "
796 "(apparently) failed.\n",
800 "%s: WARNING: deletion of output file suppressed\n",
803 "%s: since input file no longer exists. Output file\n",
806 "%s: `%s' may be incomplete.\n",
809 "%s: I suggest doing an integrity test (bzip2 -tv)"
815 if (noisy
&& numFileNames
> 0 && numFilesProcessed
< numFileNames
) {
817 "%s: WARNING: some files have not been processed:\n"
818 "%s: %d specified on command line, %d not processed yet.\n\n",
820 numFileNames
, numFileNames
- numFilesProcessed
);
827 /*---------------------------------------------*/
829 void panic ( Char
* s
)
832 "\n%s: PANIC -- internal consistency error:\n"
834 "\tThis is a BUG. Please report it to me at:\n"
835 "\tjseward@bzip.org\n",
842 /*---------------------------------------------*/
844 void crcError ( void )
847 "\n%s: Data integrity error when decompressing.\n",
855 /*---------------------------------------------*/
857 void compressedStreamEOF ( void )
861 "\n%s: Compressed file ends unexpectedly;\n\t"
862 "perhaps it is corrupted? *Possible* reason follows.\n",
872 /*---------------------------------------------*/
874 void ioError ( void )
877 "\n%s: I/O or other error, bailing out. "
878 "Possible reason follows.\n",
886 /*---------------------------------------------*/
888 void mySignalCatcher ( IntNative n
)
891 "\n%s: Control-C or similar caught, quitting.\n",
897 /*---------------------------------------------*/
899 void mySIGSEGVorSIGBUScatcher ( IntNative n
)
904 "\n%s: Caught a SIGSEGV or SIGBUS whilst compressing.\n"
906 " Possible causes are (most likely first):\n"
907 " (1) This computer has unreliable memory or cache hardware\n"
908 " (a surprisingly common problem; try a different machine.)\n"
909 " (2) A bug in the compiler used to create this executable\n"
910 " (unlikely, if you didn't compile bzip2 yourself.)\n"
911 " (3) A real bug in bzip2 -- I hope this should never be the case.\n"
912 " The user's manual, Section 4.3, has more info on (1) and (2).\n"
914 " If you suspect this is a bug in bzip2, or are unsure about (1)\n"
915 " or (2), feel free to report it to me at: jseward@bzip.org.\n"
916 " Section 4.3 of the user's manual describes the info a useful\n"
917 " bug report should have. If the manual is available on your\n"
918 " system, please try and read it before mailing me. If you don't\n"
919 " have the manual or can't be bothered to read it, mail me anyway.\n"
925 "\n%s: Caught a SIGSEGV or SIGBUS whilst decompressing.\n"
927 " Possible causes are (most likely first):\n"
928 " (1) The compressed data is corrupted, and bzip2's usual checks\n"
929 " failed to detect this. Try bzip2 -tvv my_file.bz2.\n"
930 " (2) This computer has unreliable memory or cache hardware\n"
931 " (a surprisingly common problem; try a different machine.)\n"
932 " (3) A bug in the compiler used to create this executable\n"
933 " (unlikely, if you didn't compile bzip2 yourself.)\n"
934 " (4) A real bug in bzip2 -- I hope this should never be the case.\n"
935 " The user's manual, Section 4.3, has more info on (2) and (3).\n"
937 " If you suspect this is a bug in bzip2, or are unsure about (2)\n"
938 " or (3), feel free to report it to me at: jseward@bzip.org.\n"
939 " Section 4.3 of the user's manual describes the info a useful\n"
940 " bug report should have. If the manual is available on your\n"
941 " system, please try and read it before mailing me. If you don't\n"
942 " have the manual or can't be bothered to read it, mail me anyway.\n"
948 cleanUpAndFail( 3 ); else
949 { cadvise(); cleanUpAndFail( 2 ); }
953 /*---------------------------------------------*/
955 void outOfMemory ( void )
958 "\n%s: couldn't allocate enough memory\n",
965 /*---------------------------------------------*/
967 void configError ( void )
970 "bzip2: I'm not configured correctly for this platform!\n"
971 "\tI require Int32, Int16 and Char to have sizes\n"
972 "\tof 4, 2 and 1 bytes to run properly, and they don't.\n"
973 "\tProbably you can fix this by defining them correctly,\n"
974 "\tand recompiling. Bye!\n" );
980 /*---------------------------------------------------*/
981 /*--- The main driver machinery ---*/
982 /*---------------------------------------------------*/
984 /* All rather crufty. The main problem is that input files
985 are stat()d multiple times before use. This should be
989 /*---------------------------------------------*/
994 if ( (Int32
)strlen(s
) >= longestFileName
) return;
995 for (i
= 1; i
<= longestFileName
- (Int32
)strlen(s
); i
++)
996 fprintf ( stderr
, " " );
1000 /*---------------------------------------------*/
1002 void copyFileName ( Char
* to
, Char
* from
)
1004 if ( strlen(from
) > FILE_NAME_LEN
-10 ) {
1007 "bzip2: file name\n`%s'\n"
1008 "is suspiciously (more than %d chars) long.\n"
1009 "Try using a reasonable file name instead. Sorry! :-)\n",
1010 from
, FILE_NAME_LEN
-10
1016 strncpy(to
,from
,FILE_NAME_LEN
-10);
1017 to
[FILE_NAME_LEN
-10]='\0';
1021 /*---------------------------------------------*/
1023 Bool
fileExists ( Char
* name
)
1025 FILE *tmp
= fopen ( name
, "rb" );
1026 Bool exists
= (tmp
!= NULL
);
1027 if (tmp
!= NULL
) fclose ( tmp
);
1032 /*---------------------------------------------*/
1033 /* Open an output file safely with O_EXCL and good permissions.
1034 This avoids a race condition in versions < 1.0.2, in which
1035 the file was first opened and then had its interim permissions
1036 set safely. We instead use open() to create the file with
1037 the interim permissions required. (--- --- rw-).
1039 For non-Unix platforms, if we are not worrying about
1040 security issues, simple this simply behaves like fopen.
1042 FILE* fopen_output_safely ( Char
* name
, const char* mode
)
1047 fh
= open(name
, O_WRONLY
|O_CREAT
|O_EXCL
, S_IWUSR
|S_IRUSR
);
1048 if (fh
== -1) return NULL
;
1049 fp
= fdopen(fh
, mode
);
1050 if (fp
== NULL
) close(fh
);
1053 return fopen(name
, mode
);
1058 /*---------------------------------------------*/
1060 if in doubt, return True
1063 Bool
notAStandardFile ( Char
* name
)
1066 struct MY_STAT statBuf
;
1068 i
= MY_LSTAT ( name
, &statBuf
);
1069 if (i
!= 0) return True
;
1070 if (MY_S_ISREG(statBuf
.st_mode
)) return False
;
1075 /*---------------------------------------------*/
1077 rac 11/21/98 see if file has hard links to it
1080 Int32
countHardLinks ( Char
* name
)
1083 struct MY_STAT statBuf
;
1085 i
= MY_LSTAT ( name
, &statBuf
);
1086 if (i
!= 0) return 0;
1087 return (statBuf
.st_nlink
- 1);
1091 /*---------------------------------------------*/
1092 /* Copy modification date, access date, permissions and owner from the
1093 source to destination file. We have to copy this meta-info off
1094 into fileMetaInfo before starting to compress / decompress it,
1095 because doing it afterwards means we get the wrong access time.
1097 To complicate matters, in compress() and decompress() below, the
1098 sequence of tests preceding the call to saveInputFileMetaInfo()
1099 involves calling fileExists(), which in turn establishes its result
1100 by attempting to fopen() the file, and if successful, immediately
1101 fclose()ing it again. So we have to assume that the fopen() call
1102 does not cause the access time field to be updated.
1104 Reading of the man page for stat() (man 2 stat) on RedHat 7.2 seems
1105 to imply that merely doing open() will not affect the access time.
1106 Therefore we merely need to hope that the C library only does
1107 open() as a result of fopen(), and not any kind of read()-ahead
1110 It sounds pretty fragile to me. Whether this carries across
1111 robustly to arbitrary Unix-like platforms (or even works robustly
1112 on this one, RedHat 7.2) is unknown to me. Nevertheless ...
1116 struct MY_STAT fileMetaInfo
;
1120 void saveInputFileMetaInfo ( Char
*srcName
)
1124 /* Note use of stat here, not lstat. */
1125 retVal
= MY_STAT( srcName
, &fileMetaInfo
);
1126 ERROR_IF_NOT_ZERO ( retVal
);
1132 void applySavedMetaInfoToOutputFile ( Char
*dstName
)
1136 struct utimbuf uTimBuf
;
1138 uTimBuf
.actime
= fileMetaInfo
.st_atime
;
1139 uTimBuf
.modtime
= fileMetaInfo
.st_mtime
;
1141 retVal
= chmod ( dstName
, fileMetaInfo
.st_mode
);
1142 ERROR_IF_NOT_ZERO ( retVal
);
1144 retVal
= utime ( dstName
, &uTimBuf
);
1145 ERROR_IF_NOT_ZERO ( retVal
);
1147 retVal
= chown ( dstName
, fileMetaInfo
.st_uid
, fileMetaInfo
.st_gid
);
1148 /* chown() will in many cases return with EPERM, which can
1155 /*---------------------------------------------*/
1157 Bool
containsDubiousChars ( Char
* name
)
1160 /* On unix, files can contain any characters and the file expansion
1161 * is performed by the shell.
1164 # else /* ! BZ_UNIX */
1165 /* On non-unix (Win* platforms), wildcard characters are not allowed in
1168 for (; *name
!= '\0'; name
++)
1169 if (*name
== '?' || *name
== '*') return True
;
1171 # endif /* BZ_UNIX */
1175 /*---------------------------------------------*/
1176 #define BZ_N_SUFFIX_PAIRS 4
1178 Char
* zSuffix
[BZ_N_SUFFIX_PAIRS
]
1179 = { ".bz2", ".bz", ".tbz2", ".tbz" };
1180 Char
* unzSuffix
[BZ_N_SUFFIX_PAIRS
]
1181 = { "", "", ".tar", ".tar" };
1184 Bool
hasSuffix ( Char
* s
, Char
* suffix
)
1186 Int32 ns
= strlen(s
);
1187 Int32 nx
= strlen(suffix
);
1188 if (ns
< nx
) return False
;
1189 if (strcmp(s
+ ns
- nx
, suffix
) == 0) return True
;
1194 Bool
mapSuffix ( Char
* name
,
1195 Char
* oldSuffix
, Char
* newSuffix
)
1197 if (!hasSuffix(name
,oldSuffix
)) return False
;
1198 name
[strlen(name
)-strlen(oldSuffix
)] = 0;
1199 strcat ( name
, newSuffix
);
1204 /*---------------------------------------------*/
1206 void compress ( Char
*name
)
1211 struct MY_STAT statBuf
;
1213 deleteOutputOnInterrupt
= False
;
1215 if (name
== NULL
&& srcMode
!= SM_I2O
)
1216 panic ( "compress: bad modes\n" );
1220 copyFileName ( inName
, "(stdin)" );
1221 copyFileName ( outName
, "(stdout)" );
1224 copyFileName ( inName
, name
);
1225 copyFileName ( outName
, name
);
1226 strcat ( outName
, ".bz2" );
1229 copyFileName ( inName
, name
);
1230 copyFileName ( outName
, "(stdout)" );
1234 if ( srcMode
!= SM_I2O
&& containsDubiousChars ( inName
) ) {
1236 fprintf ( stderr
, "%s: There are no files matching `%s'.\n",
1241 if ( srcMode
!= SM_I2O
&& !fileExists ( inName
) ) {
1242 fprintf ( stderr
, "%s: Can't open input file %s: %s.\n",
1243 progName
, inName
, strerror(errno
) );
1247 for (i
= 0; i
< BZ_N_SUFFIX_PAIRS
; i
++) {
1248 if (hasSuffix(inName
, zSuffix
[i
])) {
1251 "%s: Input file %s already has %s suffix.\n",
1252 progName
, inName
, zSuffix
[i
] );
1257 if ( srcMode
== SM_F2F
|| srcMode
== SM_F2O
) {
1258 MY_STAT(inName
, &statBuf
);
1259 if ( MY_S_ISDIR(statBuf
.st_mode
) ) {
1261 "%s: Input file %s is a directory.\n",
1267 if ( srcMode
== SM_F2F
&& !forceOverwrite
&& notAStandardFile ( inName
)) {
1269 fprintf ( stderr
, "%s: Input file %s is not a normal file.\n",
1274 if ( srcMode
== SM_F2F
&& fileExists ( outName
) ) {
1275 if (forceOverwrite
) {
1278 fprintf ( stderr
, "%s: Output file %s already exists.\n",
1279 progName
, outName
);
1284 if ( srcMode
== SM_F2F
&& !forceOverwrite
&&
1285 (n
=countHardLinks ( inName
)) > 0) {
1286 fprintf ( stderr
, "%s: Input file %s has %d other link%s.\n",
1287 progName
, inName
, n
, n
> 1 ? "s" : "" );
1292 if ( srcMode
== SM_F2F
) {
1293 /* Save the file's meta-info before we open it. Doing it later
1294 means we mess up the access times. */
1295 saveInputFileMetaInfo ( inName
);
1298 switch ( srcMode
) {
1303 if ( isatty ( fileno ( stdout
) ) ) {
1305 "%s: I won't write compressed data to a terminal.\n",
1307 fprintf ( stderr
, "%s: For help, type: `%s --help'.\n",
1308 progName
, progName
);
1315 inStr
= fopen ( inName
, "rb" );
1317 if ( isatty ( fileno ( stdout
) ) ) {
1319 "%s: I won't write compressed data to a terminal.\n",
1321 fprintf ( stderr
, "%s: For help, type: `%s --help'.\n",
1322 progName
, progName
);
1323 if ( inStr
!= NULL
) fclose ( inStr
);
1327 if ( inStr
== NULL
) {
1328 fprintf ( stderr
, "%s: Can't open input file %s: %s.\n",
1329 progName
, inName
, strerror(errno
) );
1336 inStr
= fopen ( inName
, "rb" );
1337 outStr
= fopen_output_safely ( outName
, "wb" );
1338 if ( outStr
== NULL
) {
1339 fprintf ( stderr
, "%s: Can't create output file %s: %s.\n",
1340 progName
, outName
, strerror(errno
) );
1341 if ( inStr
!= NULL
) fclose ( inStr
);
1345 if ( inStr
== NULL
) {
1346 fprintf ( stderr
, "%s: Can't open input file %s: %s.\n",
1347 progName
, inName
, strerror(errno
) );
1348 if ( outStr
!= NULL
) fclose ( outStr
);
1355 panic ( "compress: bad srcMode" );
1359 if (verbosity
>= 1) {
1360 fprintf ( stderr
, " %s: ", inName
);
1365 /*--- Now the input and output handles are sane. Do the Biz. ---*/
1366 outputHandleJustInCase
= outStr
;
1367 deleteOutputOnInterrupt
= True
;
1368 compressStream ( inStr
, outStr
);
1369 outputHandleJustInCase
= NULL
;
1371 /*--- If there was an I/O error, we won't get here. ---*/
1372 if ( srcMode
== SM_F2F
) {
1373 applySavedMetaInfoToOutputFile ( outName
);
1374 deleteOutputOnInterrupt
= False
;
1375 if ( !keepInputFiles
) {
1376 IntNative retVal
= remove ( inName
);
1377 ERROR_IF_NOT_ZERO ( retVal
);
1381 deleteOutputOnInterrupt
= False
;
1385 /*---------------------------------------------*/
1387 void uncompress ( Char
*name
)
1394 struct MY_STAT statBuf
;
1396 deleteOutputOnInterrupt
= False
;
1398 if (name
== NULL
&& srcMode
!= SM_I2O
)
1399 panic ( "uncompress: bad modes\n" );
1404 copyFileName ( inName
, "(stdin)" );
1405 copyFileName ( outName
, "(stdout)" );
1408 copyFileName ( inName
, name
);
1409 copyFileName ( outName
, name
);
1410 for (i
= 0; i
< BZ_N_SUFFIX_PAIRS
; i
++)
1411 if (mapSuffix(outName
,zSuffix
[i
],unzSuffix
[i
]))
1414 strcat ( outName
, ".out" );
1417 copyFileName ( inName
, name
);
1418 copyFileName ( outName
, "(stdout)" );
1423 if ( srcMode
!= SM_I2O
&& containsDubiousChars ( inName
) ) {
1425 fprintf ( stderr
, "%s: There are no files matching `%s'.\n",
1430 if ( srcMode
!= SM_I2O
&& !fileExists ( inName
) ) {
1431 fprintf ( stderr
, "%s: Can't open input file %s: %s.\n",
1432 progName
, inName
, strerror(errno
) );
1436 if ( srcMode
== SM_F2F
|| srcMode
== SM_F2O
) {
1437 MY_STAT(inName
, &statBuf
);
1438 if ( MY_S_ISDIR(statBuf
.st_mode
) ) {
1440 "%s: Input file %s is a directory.\n",
1446 if ( srcMode
== SM_F2F
&& !forceOverwrite
&& notAStandardFile ( inName
)) {
1448 fprintf ( stderr
, "%s: Input file %s is not a normal file.\n",
1453 if ( /* srcMode == SM_F2F implied && */ cantGuess
) {
1456 "%s: Can't guess original name for %s -- using %s\n",
1457 progName
, inName
, outName
);
1458 /* just a warning, no return */
1460 if ( srcMode
== SM_F2F
&& fileExists ( outName
) ) {
1461 if (forceOverwrite
) {
1464 fprintf ( stderr
, "%s: Output file %s already exists.\n",
1465 progName
, outName
);
1470 if ( srcMode
== SM_F2F
&& !forceOverwrite
&&
1471 (n
=countHardLinks ( inName
) ) > 0) {
1472 fprintf ( stderr
, "%s: Input file %s has %d other link%s.\n",
1473 progName
, inName
, n
, n
> 1 ? "s" : "" );
1478 if ( srcMode
== SM_F2F
) {
1479 /* Save the file's meta-info before we open it. Doing it later
1480 means we mess up the access times. */
1481 saveInputFileMetaInfo ( inName
);
1484 switch ( srcMode
) {
1489 if ( isatty ( fileno ( stdin
) ) ) {
1491 "%s: I won't read compressed data from a terminal.\n",
1493 fprintf ( stderr
, "%s: For help, type: `%s --help'.\n",
1494 progName
, progName
);
1501 inStr
= fopen ( inName
, "rb" );
1503 if ( inStr
== NULL
) {
1504 fprintf ( stderr
, "%s: Can't open input file %s:%s.\n",
1505 progName
, inName
, strerror(errno
) );
1506 if ( inStr
!= NULL
) fclose ( inStr
);
1513 inStr
= fopen ( inName
, "rb" );
1514 outStr
= fopen_output_safely ( outName
, "wb" );
1515 if ( outStr
== NULL
) {
1516 fprintf ( stderr
, "%s: Can't create output file %s: %s.\n",
1517 progName
, outName
, strerror(errno
) );
1518 if ( inStr
!= NULL
) fclose ( inStr
);
1522 if ( inStr
== NULL
) {
1523 fprintf ( stderr
, "%s: Can't open input file %s: %s.\n",
1524 progName
, inName
, strerror(errno
) );
1525 if ( outStr
!= NULL
) fclose ( outStr
);
1532 panic ( "uncompress: bad srcMode" );
1536 if (verbosity
>= 1) {
1537 fprintf ( stderr
, " %s: ", inName
);
1542 /*--- Now the input and output handles are sane. Do the Biz. ---*/
1543 outputHandleJustInCase
= outStr
;
1544 deleteOutputOnInterrupt
= True
;
1545 magicNumberOK
= uncompressStream ( inStr
, outStr
);
1546 outputHandleJustInCase
= NULL
;
1548 /*--- If there was an I/O error, we won't get here. ---*/
1549 if ( magicNumberOK
) {
1550 if ( srcMode
== SM_F2F
) {
1551 applySavedMetaInfoToOutputFile ( outName
);
1552 deleteOutputOnInterrupt
= False
;
1553 if ( !keepInputFiles
) {
1554 IntNative retVal
= remove ( inName
);
1555 ERROR_IF_NOT_ZERO ( retVal
);
1559 unzFailsExist
= True
;
1560 deleteOutputOnInterrupt
= False
;
1561 if ( srcMode
== SM_F2F
) {
1562 IntNative retVal
= remove ( outName
);
1563 ERROR_IF_NOT_ZERO ( retVal
);
1566 deleteOutputOnInterrupt
= False
;
1568 if ( magicNumberOK
) {
1570 fprintf ( stderr
, "done\n" );
1574 fprintf ( stderr
, "not a bzip2 file.\n" ); else
1576 "%s: %s is not a bzip2 file.\n",
1583 /*---------------------------------------------*/
1585 void testf ( Char
*name
)
1589 struct MY_STAT statBuf
;
1591 deleteOutputOnInterrupt
= False
;
1593 if (name
== NULL
&& srcMode
!= SM_I2O
)
1594 panic ( "testf: bad modes\n" );
1596 copyFileName ( outName
, "(none)" );
1598 case SM_I2O
: copyFileName ( inName
, "(stdin)" ); break;
1599 case SM_F2F
: copyFileName ( inName
, name
); break;
1600 case SM_F2O
: copyFileName ( inName
, name
); break;
1603 if ( srcMode
!= SM_I2O
&& containsDubiousChars ( inName
) ) {
1605 fprintf ( stderr
, "%s: There are no files matching `%s'.\n",
1610 if ( srcMode
!= SM_I2O
&& !fileExists ( inName
) ) {
1611 fprintf ( stderr
, "%s: Can't open input %s: %s.\n",
1612 progName
, inName
, strerror(errno
) );
1616 if ( srcMode
!= SM_I2O
) {
1617 MY_STAT(inName
, &statBuf
);
1618 if ( MY_S_ISDIR(statBuf
.st_mode
) ) {
1620 "%s: Input file %s is a directory.\n",
1627 switch ( srcMode
) {
1630 if ( isatty ( fileno ( stdin
) ) ) {
1632 "%s: I won't read compressed data from a terminal.\n",
1634 fprintf ( stderr
, "%s: For help, type: `%s --help'.\n",
1635 progName
, progName
);
1642 case SM_F2O
: case SM_F2F
:
1643 inStr
= fopen ( inName
, "rb" );
1644 if ( inStr
== NULL
) {
1645 fprintf ( stderr
, "%s: Can't open input file %s:%s.\n",
1646 progName
, inName
, strerror(errno
) );
1653 panic ( "testf: bad srcMode" );
1657 if (verbosity
>= 1) {
1658 fprintf ( stderr
, " %s: ", inName
);
1663 /*--- Now the input handle is sane. Do the Biz. ---*/
1664 outputHandleJustInCase
= NULL
;
1665 allOK
= testStream ( inStr
);
1667 if (allOK
&& verbosity
>= 1) fprintf ( stderr
, "ok\n" );
1668 if (!allOK
) testFailsExist
= True
;
1672 /*---------------------------------------------*/
1674 void license ( void )
1678 "bzip2, a block-sorting file compressor. "
1681 " Copyright (C) 1996-2005 by Julian Seward.\n"
1683 " This program is free software; you can redistribute it and/or modify\n"
1684 " it under the terms set out in the LICENSE file, which is included\n"
1685 " in the bzip2-1.0 source distribution.\n"
1687 " This program is distributed in the hope that it will be useful,\n"
1688 " but WITHOUT ANY WARRANTY; without even the implied warranty of\n"
1689 " MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n"
1690 " LICENSE file for more details.\n"
1697 /*---------------------------------------------*/
1699 void usage ( Char
*fullProgName
)
1703 "bzip2, a block-sorting file compressor. "
1705 "\n usage: %s [flags and input files in any order]\n"
1707 " -h --help print this message\n"
1708 " -d --decompress force decompression\n"
1709 " -z --compress force compression\n"
1710 " -k --keep keep (don't delete) input files\n"
1711 " -f --force overwrite existing output files\n"
1712 " -t --test test compressed file integrity\n"
1713 " -c --stdout output to standard out\n"
1714 " -q --quiet suppress noncritical error messages\n"
1715 " -v --verbose be verbose (a 2nd -v gives more)\n"
1716 " -L --license display software version & license\n"
1717 " -V --version display software version & license\n"
1718 " -s --small use less memory (at most 2500k)\n"
1719 " -1 .. -9 set block size to 100k .. 900k\n"
1720 " --fast alias for -1\n"
1721 " --best alias for -9\n"
1723 " If invoked as `bzip2', default action is to compress.\n"
1724 " as `bunzip2', default action is to decompress.\n"
1725 " as `bzcat', default action is to decompress to stdout.\n"
1727 " If no file names are given, bzip2 compresses or decompresses\n"
1728 " from standard input to standard output. You can combine\n"
1729 " short flags, so `-v -4' means the same as -v4 or -4v, &c.\n"
1741 /*---------------------------------------------*/
1743 void redundant ( Char
* flag
)
1747 "%s: %s is redundant in versions 0.9.5 and above\n",
1752 /*---------------------------------------------*/
1754 All the garbage from here to main() is purely to
1755 implement a linked list of command-line arguments,
1756 into which main() copies argv[1 .. argc-1].
1758 The purpose of this exercise is to facilitate
1759 the expansion of wildcard characters * and ? in
1760 filenames for OSs which don't know how to do it
1761 themselves, like MSDOS, Windows 95 and NT.
1763 The actual Dirty Work is done by the platform-
1764 specific macro APPEND_FILESPEC.
1775 /*---------------------------------------------*/
1777 void *myMalloc ( Int32 n
)
1781 p
= malloc ( (size_t)n
);
1782 if (p
== NULL
) outOfMemory ();
1787 /*---------------------------------------------*/
1789 Cell
*mkCell ( void )
1793 c
= (Cell
*) myMalloc ( sizeof ( Cell
) );
1800 /*---------------------------------------------*/
1802 Cell
*snocString ( Cell
*root
, Char
*name
)
1805 Cell
*tmp
= mkCell();
1806 tmp
->name
= (Char
*) myMalloc ( 5 + strlen(name
) );
1807 strcpy ( tmp
->name
, name
);
1811 while (tmp
->link
!= NULL
) tmp
= tmp
->link
;
1812 tmp
->link
= snocString ( tmp
->link
, name
);
1818 /*---------------------------------------------*/
1820 void addFlagsFromEnvVar ( Cell
** argList
, Char
* varName
)
1825 envbase
= getenv(varName
);
1826 if (envbase
!= NULL
) {
1830 if (p
[i
] == 0) break;
1833 while (isspace((Int32
)(p
[0]))) p
++;
1834 while (p
[i
] != 0 && !isspace((Int32
)(p
[i
]))) i
++;
1836 k
= i
; if (k
> FILE_NAME_LEN
-10) k
= FILE_NAME_LEN
-10;
1837 for (j
= 0; j
< k
; j
++) tmpName
[j
] = p
[j
];
1839 APPEND_FLAG(*argList
, tmpName
);
1846 /*---------------------------------------------*/
1847 #define ISFLAG(s) (strcmp(aa->name, (s))==0)
1849 IntNative
main ( IntNative argc
, Char
*argv
[] )
1857 /*-- Be really really really paranoid :-) --*/
1858 if (sizeof(Int32
) != 4 || sizeof(UInt32
) != 4 ||
1859 sizeof(Int16
) != 2 || sizeof(UInt16
) != 2 ||
1860 sizeof(Char
) != 1 || sizeof(UChar
) != 1)
1863 /*-- Initialise --*/
1864 outputHandleJustInCase
= NULL
;
1866 keepInputFiles
= False
;
1867 forceOverwrite
= False
;
1871 testFailsExist
= False
;
1872 unzFailsExist
= False
;
1874 numFilesProcessed
= 0;
1876 deleteOutputOnInterrupt
= False
;
1878 i
= j
= 0; /* avoid bogus warning from egcs-1.1.X */
1880 /*-- Set up signal handlers for mem access errors --*/
1881 signal (SIGSEGV
, mySIGSEGVorSIGBUScatcher
);
1884 signal (SIGBUS
, mySIGSEGVorSIGBUScatcher
);
1888 copyFileName ( inName
, "(none)" );
1889 copyFileName ( outName
, "(none)" );
1891 copyFileName ( progNameReally
, argv
[0] );
1892 progName
= &progNameReally
[0];
1893 for (tmp
= &progNameReally
[0]; *tmp
!= '\0'; tmp
++)
1894 if (*tmp
== PATH_SEP
) progName
= tmp
+ 1;
1897 /*-- Copy flags from env var BZIP2, and
1898 expand filename wildcards in arg list.
1901 addFlagsFromEnvVar ( &argList
, "BZIP2" );
1902 addFlagsFromEnvVar ( &argList
, "BZIP" );
1903 for (i
= 1; i
<= argc
-1; i
++)
1904 APPEND_FILESPEC(argList
, argv
[i
]);
1907 /*-- Find the length of the longest filename --*/
1908 longestFileName
= 7;
1911 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
1912 if (ISFLAG("--")) { decode
= False
; continue; }
1913 if (aa
->name
[0] == '-' && decode
) continue;
1915 if (longestFileName
< (Int32
)strlen(aa
->name
) )
1916 longestFileName
= (Int32
)strlen(aa
->name
);
1920 /*-- Determine source modes; flag handling may change this too. --*/
1921 if (numFileNames
== 0)
1922 srcMode
= SM_I2O
; else srcMode
= SM_F2F
;
1925 /*-- Determine what to do (compress/uncompress/test/cat). --*/
1926 /*-- Note that subsequent flag handling may change this. --*/
1929 if ( (strstr ( progName
, "unzip" ) != 0) ||
1930 (strstr ( progName
, "UNZIP" ) != 0) )
1933 if ( (strstr ( progName
, "z2cat" ) != 0) ||
1934 (strstr ( progName
, "Z2CAT" ) != 0) ||
1935 (strstr ( progName
, "zcat" ) != 0) ||
1936 (strstr ( progName
, "ZCAT" ) != 0) ) {
1938 srcMode
= (numFileNames
== 0) ? SM_I2O
: SM_F2O
;
1941 if(strstr ( progName
, "small" ) != 0) {
1946 /*-- Look at the flags. --*/
1947 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
1948 if (ISFLAG("--")) break;
1949 if (aa
->name
[0] == '-' && aa
->name
[1] != '-') {
1950 for (j
= 1; aa
->name
[j
] != '\0'; j
++) {
1951 switch (aa
->name
[j
]) {
1952 case 'c': srcMode
= SM_F2O
; break;
1953 case 'd': opMode
= OM_UNZ
; break;
1954 case 'z': opMode
= OM_Z
; break;
1955 case 'f': forceOverwrite
= True
; break;
1956 case 't': opMode
= OM_TEST
; break;
1957 case 'k': keepInputFiles
= True
; break;
1958 case 's': smallMode
= True
; break;
1959 case 'q': noisy
= False
; break;
1960 case '1': blockSize100k
= 1; break;
1961 case '2': blockSize100k
= 2; break;
1962 case '3': blockSize100k
= 3; break;
1963 case '4': blockSize100k
= 4; break;
1964 case '5': blockSize100k
= 5; break;
1965 case '6': blockSize100k
= 6; break;
1966 case '7': blockSize100k
= 7; break;
1967 case '8': blockSize100k
= 8; break;
1968 case '9': blockSize100k
= 9; break;
1970 case 'L': license(); break;
1971 case 'v': verbosity
++; break;
1972 case 'h': usage ( progName
);
1975 default: fprintf ( stderr
, "%s: Bad flag `%s'\n",
1976 progName
, aa
->name
);
1985 /*-- And again ... --*/
1986 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
1987 if (ISFLAG("--")) break;
1988 if (ISFLAG("--stdout")) srcMode
= SM_F2O
; else
1989 if (ISFLAG("--decompress")) opMode
= OM_UNZ
; else
1990 if (ISFLAG("--compress")) opMode
= OM_Z
; else
1991 if (ISFLAG("--force")) forceOverwrite
= True
; else
1992 if (ISFLAG("--test")) opMode
= OM_TEST
; else
1993 if (ISFLAG("--keep")) keepInputFiles
= True
; else
1994 if (ISFLAG("--small")) smallMode
= True
; else
1995 if (ISFLAG("--quiet")) noisy
= False
; else
1996 if (ISFLAG("--version")) license(); else
1997 if (ISFLAG("--license")) license(); else
1998 if (ISFLAG("--exponential")) workFactor
= 1; else
1999 if (ISFLAG("--repetitive-best")) redundant(aa
->name
); else
2000 if (ISFLAG("--repetitive-fast")) redundant(aa
->name
); else
2001 if (ISFLAG("--fast")) blockSize100k
= 1; else
2002 if (ISFLAG("--best")) blockSize100k
= 9; else
2003 if (ISFLAG("--verbose")) verbosity
++; else
2004 if (ISFLAG("--help")) { usage ( progName
); exit ( 0 ); }
2006 if (strncmp ( aa
->name
, "--", 2) == 0) {
2007 fprintf ( stderr
, "%s: Bad flag `%s'\n", progName
, aa
->name
);
2013 if (verbosity
> 4) verbosity
= 4;
2014 if (opMode
== OM_Z
&& smallMode
&& blockSize100k
> 2)
2017 if (opMode
== OM_TEST
&& srcMode
== SM_F2O
) {
2018 fprintf ( stderr
, "%s: -c and -t cannot be used together.\n",
2023 if (srcMode
== SM_F2O
&& numFileNames
== 0)
2026 if (opMode
!= OM_Z
) blockSize100k
= 0;
2028 if (srcMode
== SM_F2F
) {
2029 signal (SIGINT
, mySignalCatcher
);
2030 signal (SIGTERM
, mySignalCatcher
);
2032 signal (SIGHUP
, mySignalCatcher
);
2036 if (opMode
== OM_Z
) {
2037 if (srcMode
== SM_I2O
) {
2041 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
2042 if (ISFLAG("--")) { decode
= False
; continue; }
2043 if (aa
->name
[0] == '-' && decode
) continue;
2044 numFilesProcessed
++;
2045 compress ( aa
->name
);
2051 if (opMode
== OM_UNZ
) {
2052 unzFailsExist
= False
;
2053 if (srcMode
== SM_I2O
) {
2054 uncompress ( NULL
);
2057 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
2058 if (ISFLAG("--")) { decode
= False
; continue; }
2059 if (aa
->name
[0] == '-' && decode
) continue;
2060 numFilesProcessed
++;
2061 uncompress ( aa
->name
);
2064 if (unzFailsExist
) {
2071 testFailsExist
= False
;
2072 if (srcMode
== SM_I2O
) {
2076 for (aa
= argList
; aa
!= NULL
; aa
= aa
->link
) {
2077 if (ISFLAG("--")) { decode
= False
; continue; }
2078 if (aa
->name
[0] == '-' && decode
) continue;
2079 numFilesProcessed
++;
2083 if (testFailsExist
&& noisy
) {
2086 "You can use the `bzip2recover' program to attempt to recover\n"
2087 "data from undamaged sections of corrupted files.\n\n"
2094 /* Free the argument list memory to mollify leak detectors
2095 (eg) Purify, Checker. Serves no other useful purpose.
2098 while (aa
!= NULL
) {
2099 Cell
* aa2
= aa
->link
;
2100 if (aa
->name
!= NULL
) free(aa
->name
);
2109 /*-----------------------------------------------------------*/
2110 /*--- end bzip2.c ---*/
2111 /*-----------------------------------------------------------*/