bzip2.c
上传用户:zswatin
上传日期:2007-01-06
资源大小:440k
文件大小:45k
源码类别:

压缩解压

开发平台:

C/C++

  1. /*-----------------------------------------------------------*/
  2. /*--- A block-sorting, lossless compressor        bzip2.c ---*/
  3. /*-----------------------------------------------------------*/
  4. /*--
  5.   This file is a part of bzip2 and/or libbzip2, a program and
  6.   library for lossless, block-sorting data compression.
  7.   Copyright (C) 1996-1998 Julian R Seward.  All rights reserved.
  8.   Redistribution and use in source and binary forms, with or without
  9.   modification, are permitted provided that the following conditions
  10.   are met:
  11.   1. Redistributions of source code must retain the above copyright
  12.      notice, this list of conditions and the following disclaimer.
  13.   2. The origin of this software must not be misrepresented; you must 
  14.      not claim that you wrote the original software.  If you use this 
  15.      software in a product, an acknowledgment in the product 
  16.      documentation would be appreciated but is not required.
  17.   3. Altered source versions must be plainly marked as such, and must
  18.      not be misrepresented as being the original software.
  19.   4. The name of the author may not be used to endorse or promote 
  20.      products derived from this software without specific prior written 
  21.      permission.
  22.   THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
  23.   OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  24.   WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  25.   ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  26.   DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  27.   DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
  28.   GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  29.   INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
  30.   WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  31.   NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  32.   SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  33.   Julian Seward, Guildford, Surrey, UK.
  34.   jseward@acm.org
  35.   bzip2/libbzip2 version 0.9.0c of 18 October 1998
  36.   This program is based on (at least) the work of:
  37.      Mike Burrows
  38.      David Wheeler
  39.      Peter Fenwick
  40.      Alistair Moffat
  41.      Radford Neal
  42.      Ian H. Witten
  43.      Robert Sedgewick
  44.      Jon L. Bentley
  45.   For more information on these sources, see the manual.
  46. --*/
  47. /*----------------------------------------------------*/
  48. /*--- IMPORTANT                                    ---*/
  49. /*----------------------------------------------------*/
  50. /*--
  51.    WARNING:
  52.       This program and library (attempts to) compress data by 
  53.       performing several non-trivial transformations on it.  
  54.       Unless you are 100% familiar with *all* the algorithms 
  55.       contained herein, and with the consequences of modifying them, 
  56.       you should NOT meddle with the compression or decompression 
  57.       machinery.  Incorrect changes can and very likely *will* 
  58.       lead to disasterous loss of data.
  59.    DISCLAIMER:
  60.       I TAKE NO RESPONSIBILITY FOR ANY LOSS OF DATA ARISING FROM THE
  61.       USE OF THIS PROGRAM, HOWSOEVER CAUSED.
  62.       Every compression of a file implies an assumption that the
  63.       compressed file can be decompressed to reproduce the original.
  64.       Great efforts in design, coding and testing have been made to
  65.       ensure that this program works correctly.  However, the
  66.       complexity of the algorithms, and, in particular, the presence
  67.       of various special cases in the code which occur with very low
  68.       but non-zero probability make it impossible to rule out the
  69.       possibility of bugs remaining in the program.  DO NOT COMPRESS
  70.       ANY DATA WITH THIS PROGRAM AND/OR LIBRARY UNLESS YOU ARE PREPARED 
  71.       TO ACCEPT THE POSSIBILITY, HOWEVER SMALL, THAT THE DATA WILL 
  72.       NOT BE RECOVERABLE.
  73.       That is not to say this program is inherently unreliable.
  74.       Indeed, I very much hope the opposite is true.  bzip2/libbzip2
  75.       has been carefully constructed and extensively tested.
  76.    PATENTS:
  77.       To the best of my knowledge, bzip2/libbzip2 does not use any 
  78.       patented algorithms.  However, I do not have the resources 
  79.       available to carry out a full patent search.  Therefore I cannot 
  80.       give any guarantee of the above statement.
  81. --*/
  82. /*----------------------------------------------------*/
  83. /*--- and now for something much more pleasant :-) ---*/
  84. /*----------------------------------------------------*/
  85. /*---------------------------------------------*/
  86. /*--
  87.   Place a 1 beside your platform, and 0 elsewhere.
  88. --*/
  89. /*--
  90.   Generic 32-bit Unix.
  91.   Also works on 64-bit Unix boxes.
  92. --*/
  93. #define BZ_UNIX      1
  94. /*--
  95.   Win32, as seen by Jacob Navia's excellent
  96.   port of (Chris Fraser & David Hanson)'s excellent
  97.   lcc compiler.
  98. --*/
  99. #define BZ_LCCWIN32  0
  100. #ifdef _WIN32
  101. #define BZ_LCCWIN32 1
  102. #define BZ_UNIX 0
  103. #endif
  104. /*---------------------------------------------*/
  105. /*--
  106.   Some stuff for all platforms.
  107. --*/
  108. #include <stdio.h>
  109. #include <stdlib.h>
  110. #include <string.h>
  111. #include <signal.h>
  112. #include <math.h>
  113. #include "bzlib.h"
  114. #define ERROR_IF_EOF(i)       { if ((i) == EOF)  ioError(); }
  115. #define ERROR_IF_NOT_ZERO(i)  { if ((i) != 0)    ioError(); }
  116. #define ERROR_IF_MINUS_ONE(i) { if ((i) == (-1)) ioError(); }
  117. /*---------------------------------------------*/
  118. /*--
  119.    Platform-specific stuff.
  120. --*/
  121. #if BZ_UNIX
  122. #   include <sys/types.h>
  123. #   include <utime.h>
  124. #   include <unistd.h>
  125. #   include <sys/stat.h>
  126. #   include <sys/times.h>
  127. #   define PATH_SEP    '/'
  128. #   define MY_LSTAT    lstat
  129. #   define MY_S_IFREG  S_ISREG
  130. #   define MY_STAT     stat
  131. #   define APPEND_FILESPEC(root, name) 
  132.       root=snocString((root), (name))
  133. #   define SET_BINARY_MODE(fd) /**/
  134. #   ifdef __GNUC__
  135. #      define NORETURN __attribute__ ((noreturn))
  136. #   else
  137. #      define NORETURN /**/
  138. #   endif
  139. #endif
  140. #if BZ_LCCWIN32
  141. #   include <io.h>
  142. #   include <fcntl.h>
  143. #   include <sysstat.h>
  144. #   define NORETURN       /**/
  145. #   define PATH_SEP       '\'
  146. #   define MY_LSTAT       _stat
  147. #   define MY_STAT        _stat
  148. #   define MY_S_IFREG(x)  ((x) & _S_IFREG)
  149. #   if 0
  150.    /*-- lcc-win32 seems to expand wildcards itself --*/
  151. #   define APPEND_FILESPEC(root, spec)                
  152.       do {                                            
  153.          if ((spec)[0] == '-') {                      
  154.             root = snocString((root), (spec));        
  155.          } else {                                     
  156.             struct _finddata_t c_file;                
  157.             long hFile;                               
  158.             hFile = _findfirst((spec), &c_file);      
  159.             if ( hFile == -1L ) {                     
  160.                root = snocString ((root), (spec));    
  161.             } else {                                  
  162.                int anInt = 0;                         
  163.                while ( anInt == 0 ) {                 
  164.                   root = snocString((root),           
  165.                             &c_file.name[0]);         
  166.                   anInt = _findnext(hFile, &c_file);  
  167.                }                                      
  168.             }                                         
  169.          }                                            
  170.       } while ( 0 )
  171. #   else
  172. #   define APPEND_FILESPEC(root, name)                
  173.       root = snocString ((root), (name))
  174. #   endif
  175. #   define SET_BINARY_MODE(fd)                        
  176.       do {                                            
  177.          int retVal = setmode ( fileno ( fd ),        
  178.                                O_BINARY );            
  179.          ERROR_IF_MINUS_ONE ( retVal );               
  180.       } while ( 0 )
  181. #endif
  182. /*---------------------------------------------*/
  183. /*--
  184.   Some more stuff for all platforms :-)
  185. --*/
  186. typedef char            Char;
  187. typedef unsigned char   Bool;
  188. typedef unsigned char   UChar;
  189. typedef int             Int32;
  190. typedef unsigned int    UInt32;
  191. typedef short           Int16;
  192. typedef unsigned short  UInt16;
  193.                                        
  194. #define True  ((Bool)1)
  195. #define False ((Bool)0)
  196. /*--
  197.   IntNative is your platform's `native' int size.
  198.   Only here to avoid probs with 64-bit platforms.
  199. --*/
  200. typedef int IntNative;
  201. /*---------------------------------------------------*/
  202. /*--- Misc (file handling) data decls             ---*/
  203. /*---------------------------------------------------*/
  204. Int32   verbosity;
  205. Bool    keepInputFiles, smallMode;
  206. Bool    forceOverwrite, testFailsExist;
  207. Int32   numFileNames, numFilesProcessed, blockSize100k;
  208. /*-- source modes; F==file, I==stdin, O==stdout --*/
  209. #define SM_I2O           1
  210. #define SM_F2O           2
  211. #define SM_F2F           3
  212. /*-- operation modes --*/
  213. #define OM_Z             1
  214. #define OM_UNZ           2
  215. #define OM_TEST          3
  216. Int32   opMode;
  217. Int32   srcMode;
  218. #define FILE_NAME_LEN 1034
  219. Int32   longestFileName;
  220. Char    inName[FILE_NAME_LEN];
  221. Char    outName[FILE_NAME_LEN];
  222. Char    *progName;
  223. Char    progNameReally[FILE_NAME_LEN];
  224. FILE    *outputHandleJustInCase;
  225. Int32   workFactor;
  226. void    panic                 ( Char* )   NORETURN;
  227. void    ioError               ( void )    NORETURN;
  228. void    outOfMemory           ( void )    NORETURN;
  229. void    blockOverrun          ( void )    NORETURN;
  230. void    badBlockHeader        ( void )    NORETURN;
  231. void    badBGLengths          ( void )    NORETURN;
  232. void    crcError              ( void )    NORETURN;
  233. void    bitStreamEOF          ( void )    NORETURN;
  234. void    cleanUpAndFail        ( Int32 )   NORETURN;
  235. void    compressedStreamEOF   ( void )    NORETURN;
  236. void    copyFileName ( Char*, Char* );
  237. void*   myMalloc ( Int32 );
  238. /*---------------------------------------------------*/
  239. /*--- Processing of complete files and streams    ---*/
  240. /*---------------------------------------------------*/
  241. /*---------------------------------------------*/
  242. Bool myfeof ( FILE* f )
  243. {
  244.    Int32 c = fgetc ( f );
  245.    if (c == EOF) return True;
  246.    ungetc ( c, f );
  247.    return False;
  248. }
  249. /*---------------------------------------------*/
  250. void compressStream ( FILE *stream, FILE *zStream )
  251. {
  252.    BZFILE* bzf = NULL;
  253.    UChar   ibuf[5000];
  254.    Int32   nIbuf;
  255.    UInt32  nbytes_in, nbytes_out;
  256.    Int32   bzerr, bzerr_dummy, ret;
  257.    SET_BINARY_MODE(stream);
  258.    SET_BINARY_MODE(zStream);
  259.    if (ferror(stream)) goto errhandler_io;
  260.    if (ferror(zStream)) goto errhandler_io;
  261.    bzf = bzWriteOpen ( &bzerr, zStream, 
  262.                        blockSize100k, verbosity, workFactor );   
  263.    if (bzerr != BZ_OK) goto errhandler;
  264.    if (verbosity >= 2) fprintf ( stderr, "n" );
  265.    while (True) {
  266.       if (myfeof(stream)) break;
  267.       nIbuf = fread ( ibuf, sizeof(UChar), 5000, stream );
  268.       if (ferror(stream)) goto errhandler_io;
  269.       if (nIbuf > 0) bzWrite ( &bzerr, bzf, (void*)ibuf, nIbuf );
  270.       if (bzerr != BZ_OK) goto errhandler;
  271.    }
  272.    bzWriteClose ( &bzerr, bzf, 0, &nbytes_in, &nbytes_out );
  273.    if (bzerr != BZ_OK) goto errhandler;
  274.    if (ferror(zStream)) goto errhandler_io;
  275.    ret = fflush ( zStream );
  276.    if (ret == EOF) goto errhandler_io;
  277.    if (zStream != stdout) {
  278.       ret = fclose ( zStream );
  279.       if (ret == EOF) goto errhandler_io;
  280.    }
  281.    if (ferror(stream)) goto errhandler_io;
  282.    ret = fclose ( stream );
  283.    if (ret == EOF) goto errhandler_io;
  284.    if (nbytes_in == 0) nbytes_in = 1;
  285.    if (verbosity >= 1)
  286.       fprintf ( stderr, "%6.3f:1, %6.3f bits/byte, "
  287.                         "%5.2f%% saved, %d in, %d out.n",
  288.                 (float)nbytes_in / (float)nbytes_out,
  289.                 (8.0 * (float)nbytes_out) / (float)nbytes_in,
  290.                 100.0 * (1.0 - (float)nbytes_out / (float)nbytes_in),
  291.                 nbytes_in,
  292.                 nbytes_out
  293.               );
  294.    return;
  295.    errhandler:
  296.    bzWriteClose ( &bzerr_dummy, bzf, 1, &nbytes_in, &nbytes_out );
  297.    switch (bzerr) {
  298.       case BZ_MEM_ERROR:
  299.          outOfMemory ();
  300.       case BZ_IO_ERROR:
  301.          errhandler_io:
  302.          ioError(); break;
  303.       default:
  304.          panic ( "compress:unexpected error" );
  305.    }
  306.    panic ( "compress:end" );
  307.    /*notreached*/
  308. }
  309. /*---------------------------------------------*/
  310. Bool uncompressStream ( FILE *zStream, FILE *stream )
  311. {
  312.    BZFILE* bzf = NULL;
  313.    Int32   bzerr, bzerr_dummy, ret, nread, streamNo, i;
  314.    UChar   obuf[5000];
  315.    UChar   unused[BZ_MAX_UNUSED];
  316.    Int32   nUnused;
  317.    UChar*  unusedTmp;
  318.    nUnused = 0;
  319.    streamNo = 0;
  320.    SET_BINARY_MODE(stream);
  321.    SET_BINARY_MODE(zStream);
  322.    if (ferror(stream)) goto errhandler_io;
  323.    if (ferror(zStream)) goto errhandler_io;
  324.    while (True) {
  325.       bzf = bzReadOpen ( 
  326.                &bzerr, zStream, verbosity, 
  327.                (int)smallMode, unused, nUnused
  328.             );
  329.       if (bzf == NULL || bzerr != BZ_OK) goto errhandler;
  330.       streamNo++;
  331.       while (bzerr == BZ_OK) {
  332.          nread = bzRead ( &bzerr, bzf, obuf, 5000 );
  333.          if (bzerr == BZ_DATA_ERROR_MAGIC) goto errhandler;
  334.          if ((bzerr == BZ_OK || bzerr == BZ_STREAM_END) && nread > 0)
  335.             fwrite ( obuf, sizeof(UChar), nread, stream );
  336.          if (ferror(stream)) goto errhandler_io;
  337.       }
  338.       if (bzerr != BZ_STREAM_END) goto errhandler;
  339.       bzReadGetUnused ( &bzerr, bzf, (void**)(&unusedTmp), &nUnused );
  340.       if (bzerr != BZ_OK) panic ( "decompress:bzReadGetUnused" );
  341.       for (i = 0; i < nUnused; i++) unused[i] = unusedTmp[i];
  342.       bzReadClose ( &bzerr, bzf );
  343.       if (bzerr != BZ_OK) panic ( "decompress:bzReadGetUnused" );
  344.       if (nUnused == 0 && myfeof(zStream)) break;
  345.    }
  346.    if (ferror(zStream)) goto errhandler_io;
  347.    ret = fclose ( zStream );
  348.    if (ret == EOF) goto errhandler_io;
  349.    if (ferror(stream)) goto errhandler_io;
  350.    ret = fflush ( stream );
  351.    if (ret != 0) goto errhandler_io;
  352.    if (stream != stdout) {
  353.       ret = fclose ( stream );
  354.       if (ret == EOF) goto errhandler_io;
  355.    }
  356.    if (verbosity >= 2) fprintf ( stderr, "n    " );
  357.    return True;
  358.    errhandler:
  359.    bzReadClose ( &bzerr_dummy, bzf );
  360.    switch (bzerr) {
  361.       case BZ_IO_ERROR:
  362.          errhandler_io:
  363.          ioError(); break;
  364.       case BZ_DATA_ERROR:
  365.          crcError();
  366.       case BZ_MEM_ERROR:
  367.          outOfMemory();
  368.       case BZ_UNEXPECTED_EOF:
  369.          compressedStreamEOF();
  370.       case BZ_DATA_ERROR_MAGIC:
  371.          if (streamNo == 1) {
  372.             return False;
  373.          } else {
  374.             fprintf ( stderr, 
  375.                       "n%s: %s: trailing garbage after EOF ignoredn",
  376.                       progName, inName );
  377.             return True;       
  378.          }
  379.       default:
  380.          panic ( "decompress:unexpected error" );
  381.    }
  382.    panic ( "decompress:end" );
  383.    return True; /*notreached*/
  384. }
  385. /*---------------------------------------------*/
  386. Bool testStream ( FILE *zStream )
  387. {
  388.    BZFILE* bzf = NULL;
  389.    Int32   bzerr, bzerr_dummy, ret, nread, streamNo, i;
  390.    UChar   obuf[5000];
  391.    UChar   unused[BZ_MAX_UNUSED];
  392.    Int32   nUnused;
  393.    UChar*  unusedTmp;
  394.    nUnused = 0;
  395.    streamNo = 0;
  396.    SET_BINARY_MODE(zStream);
  397.    if (ferror(zStream)) goto errhandler_io;
  398.    while (True) {
  399.       bzf = bzReadOpen ( 
  400.                &bzerr, zStream, verbosity, 
  401.                (int)smallMode, unused, nUnused
  402.             );
  403.       if (bzf == NULL || bzerr != BZ_OK) goto errhandler;
  404.       streamNo++;
  405.       while (bzerr == BZ_OK) {
  406.          nread = bzRead ( &bzerr, bzf, obuf, 5000 );
  407.          if (bzerr == BZ_DATA_ERROR_MAGIC) goto errhandler;
  408.       }
  409.       if (bzerr != BZ_STREAM_END) goto errhandler;
  410.       bzReadGetUnused ( &bzerr, bzf, (void**)(&unusedTmp), &nUnused );
  411.       if (bzerr != BZ_OK) panic ( "test:bzReadGetUnused" );
  412.       for (i = 0; i < nUnused; i++) unused[i] = unusedTmp[i];
  413.       bzReadClose ( &bzerr, bzf );
  414.       if (bzerr != BZ_OK) panic ( "test:bzReadGetUnused" );
  415.       if (nUnused == 0 && myfeof(zStream)) break;
  416.    }
  417.    if (ferror(zStream)) goto errhandler_io;
  418.    ret = fclose ( zStream );
  419.    if (ret == EOF) goto errhandler_io;
  420.    if (verbosity >= 2) fprintf ( stderr, "n    " );
  421.    return True;
  422.    errhandler:
  423.    bzReadClose ( &bzerr_dummy, bzf );
  424.    switch (bzerr) {
  425.       case BZ_IO_ERROR:
  426.          errhandler_io:
  427.          ioError(); break;
  428.       case BZ_DATA_ERROR:
  429.          fprintf ( stderr,
  430.                    "n%s: data integrity (CRC) error in datan",
  431.                    inName );
  432.          return False;
  433.       case BZ_MEM_ERROR:
  434.          outOfMemory();
  435.       case BZ_UNEXPECTED_EOF:
  436.          fprintf ( stderr,
  437.                    "n%s: file ends unexpectedlyn",
  438.                    inName );
  439.          return False;
  440.       case BZ_DATA_ERROR_MAGIC:
  441.          if (streamNo == 1) {
  442.           fprintf ( stderr, 
  443.                     "n%s: bad magic number (ie, not created by bzip2)n",
  444.                     inName );
  445.             return False;
  446.          } else {
  447.             fprintf ( stderr, 
  448.                       "n%s: %s: trailing garbage after EOF ignoredn",
  449.                       progName, inName );
  450.             return True;       
  451.          }
  452.       default:
  453.          panic ( "test:unexpected error" );
  454.    }
  455.    panic ( "test:end" );
  456.    return True; /*notreached*/
  457. }
  458. /*---------------------------------------------------*/
  459. /*--- Error [non-] handling grunge                ---*/
  460. /*---------------------------------------------------*/
  461. /*---------------------------------------------*/
  462. void cadvise ( void )
  463. {
  464.    fprintf (
  465.       stderr,
  466.       "nIt is possible that the compressed file(s) have become corrupted.n"
  467.         "You can use the -tvv option to test integrity of such files.nn"
  468.         "You can use the `bzip2recover' program to *attempt* to recovern"
  469.         "data from undamaged sections of corrupted files.nn"
  470.     );
  471. }
  472. /*---------------------------------------------*/
  473. void showFileNames ( void )
  474. {
  475.    fprintf (
  476.       stderr,
  477.       "tInput file = %s, output file = %sn",
  478.       inName, outName 
  479.    );
  480. }
  481. /*---------------------------------------------*/
  482. void cleanUpAndFail ( Int32 ec )
  483. {
  484.    IntNative retVal;
  485.    if ( srcMode == SM_F2F && opMode != OM_TEST ) {
  486.       fprintf ( stderr, "%s: Deleting output file %s, if it exists.n",
  487.                 progName, outName );
  488.       if (outputHandleJustInCase != NULL)
  489.          fclose ( outputHandleJustInCase );
  490.       retVal = remove ( outName );
  491.       if (retVal != 0)
  492.          fprintf ( stderr,
  493.                    "%s: WARNING: deletion of output file (apparently) failed.n",
  494.                    progName );
  495.    }
  496.    if (numFileNames > 0 && numFilesProcessed < numFileNames) {
  497.       fprintf ( stderr, 
  498.                 "%s: WARNING: some files have not been processed:n"
  499.                 "t%d specified on command line, %d not processed yet.nn",
  500.                 progName, numFileNames, 
  501.                           numFileNames - numFilesProcessed );
  502.    }
  503.    exit ( ec );
  504. }
  505. /*---------------------------------------------*/
  506. void panic ( Char* s )
  507. {
  508.    fprintf ( stderr,
  509.              "n%s: PANIC -- internal consistency error:n"
  510.              "t%sn"
  511.              "tThis is a BUG.  Please report it to me at:n"
  512.              "tjseward@acm.orgn",
  513.              progName, s );
  514.    showFileNames();
  515.    cleanUpAndFail( 3 );
  516. }
  517. /*---------------------------------------------*/
  518. void crcError ()
  519. {
  520.    fprintf ( stderr,
  521.              "n%s: Data integrity error when decompressing.n",
  522.              progName );
  523.    showFileNames();
  524.    cadvise();
  525.    cleanUpAndFail( 2 );
  526. }
  527. /*---------------------------------------------*/
  528. void compressedStreamEOF ( void )
  529. {
  530.    fprintf ( stderr,
  531.              "n%s: Compressed file ends unexpectedly;nt"
  532.              "perhaps it is corrupted?  *Possible* reason follows.n",
  533.              progName );
  534.    perror ( progName );
  535.    showFileNames();
  536.    cadvise();
  537.    cleanUpAndFail( 2 );
  538. }
  539. /*---------------------------------------------*/
  540. void ioError ( )
  541. {
  542.    fprintf ( stderr,
  543.              "n%s: I/O or other error, bailing out.  Possible reason follows.n",
  544.              progName );
  545.    perror ( progName );
  546.    showFileNames();
  547.    cleanUpAndFail( 1 );
  548. }
  549. /*---------------------------------------------*/
  550. void mySignalCatcher ( IntNative n )
  551. {
  552.    fprintf ( stderr,
  553.              "n%s: Control-C (or similar) caught, quitting.n",
  554.              progName );
  555.    cleanUpAndFail(1);
  556. }
  557. /*---------------------------------------------*/
  558. void mySIGSEGVorSIGBUScatcher ( IntNative n )
  559. {
  560.    if (opMode == OM_Z)
  561.       fprintf ( stderr,
  562.                 "n%s: Caught a SIGSEGV or SIGBUS whilst compressing,n"
  563.                 "twhich probably indicates a bug in bzip2.  Pleasen"
  564.                 "treport it to me at: jseward@acm.orgn",
  565.                 progName );
  566.       else
  567.       fprintf ( stderr,
  568.                 "n%s: Caught a SIGSEGV or SIGBUS whilst decompressing,n"
  569.                 "twhich probably indicates that the compressed datan"
  570.                 "tis corrupted.n",
  571.                 progName );
  572.    showFileNames();
  573.    if (opMode == OM_Z)
  574.       cleanUpAndFail( 3 ); else
  575.       { cadvise(); cleanUpAndFail( 2 ); }
  576. }
  577. /*---------------------------------------------*/
  578. void outOfMemory ( void )
  579. {
  580.    fprintf ( stderr,
  581.              "n%s: couldn't allocate enough memoryn",
  582.              progName );
  583.    showFileNames();
  584.    cleanUpAndFail(1);
  585. }
  586. /*---------------------------------------------------*/
  587. /*--- The main driver machinery                   ---*/
  588. /*---------------------------------------------------*/
  589. /*---------------------------------------------*/
  590. void pad ( Char *s )
  591. {
  592.    Int32 i;
  593.    if ( (Int32)strlen(s) >= longestFileName ) return;
  594.    for (i = 1; i <= longestFileName - (Int32)strlen(s); i++)
  595.       fprintf ( stderr, " " );
  596. }
  597. /*---------------------------------------------*/
  598. void copyFileName ( Char* to, Char* from ) 
  599. {
  600.    if ( strlen(from) > FILE_NAME_LEN-10 )  {
  601.       fprintf (
  602.          stderr,
  603.          "bzip2: file namen`%s'nis suspiciously (> 1024 chars) long.n"
  604.          "Try using a reasonable file name instead.  Sorry! :)n",
  605.          from
  606.       );
  607.       exit(1);
  608.    }
  609.   strncpy(to,from,FILE_NAME_LEN-10);
  610.   to[FILE_NAME_LEN-10]='';
  611. }
  612. /*---------------------------------------------*/
  613. Bool fileExists ( Char* name )
  614. {
  615.    FILE *tmp   = fopen ( name, "rb" );
  616.    Bool exists = (tmp != NULL);
  617.    if (tmp != NULL) fclose ( tmp );
  618.    return exists;
  619. }
  620. /*---------------------------------------------*/
  621. /*--
  622.   if in doubt, return True
  623. --*/
  624. Bool notAStandardFile ( Char* name )
  625. {
  626.    IntNative      i;
  627.    struct MY_STAT statBuf;
  628.    i = MY_LSTAT ( name, &statBuf );
  629.    if (i != 0) return True;
  630.    if (MY_S_IFREG(statBuf.st_mode)) return False;
  631.    return True;
  632. }
  633. /*---------------------------------------------*/
  634. void copyDatePermissionsAndOwner ( Char *srcName, Char *dstName )
  635. {
  636. #if BZ_UNIX
  637.    IntNative      retVal;
  638.    struct MY_STAT statBuf;
  639.    struct utimbuf uTimBuf;
  640.    retVal = MY_LSTAT ( srcName, &statBuf );
  641.    ERROR_IF_NOT_ZERO ( retVal );
  642.    uTimBuf.actime = statBuf.st_atime;
  643.    uTimBuf.modtime = statBuf.st_mtime;
  644.    retVal = chmod ( dstName, statBuf.st_mode );
  645.    ERROR_IF_NOT_ZERO ( retVal );
  646.    /* Not sure if this is really portable or not.  Causes 
  647.       problems on my x86-Linux Redhat 5.0 box.  Decided
  648.       to omit it from 0.9.0.  JRS, 27 June 98.  If you 
  649.       understand Unix file semantics and portability issues
  650.       well enough to fix this properly, drop me a line
  651.       at jseward@acm.org.
  652.    retVal = chown ( dstName, statBuf.st_uid, statBuf.st_gid );
  653.    ERROR_IF_NOT_ZERO ( retVal );
  654.    */
  655.    retVal = utime ( dstName, &uTimBuf );
  656.    ERROR_IF_NOT_ZERO ( retVal );
  657. #endif
  658. }
  659. /*---------------------------------------------*/
  660. void setInterimPermissions ( Char *dstName )
  661. {
  662. #if BZ_UNIX
  663.    IntNative      retVal;
  664.    retVal = chmod ( dstName, S_IRUSR | S_IWUSR );
  665.    ERROR_IF_NOT_ZERO ( retVal );
  666. #endif
  667. }
  668. /*---------------------------------------------*/
  669. Bool endsInBz2 ( Char* name )
  670. {
  671.    Int32 n = strlen ( name );
  672.    if (n <= 4) return False;
  673.    return
  674.       (name[n-4] == '.' &&
  675.        name[n-3] == 'b' &&
  676.        name[n-2] == 'z' &&
  677.        name[n-1] == '2');
  678. }
  679. /*---------------------------------------------*/
  680. Bool containsDubiousChars ( Char* name )
  681. {
  682.    Bool cdc = False;
  683.    for (; *name != ''; name++)
  684.       if (*name == '?' || *name == '*') cdc = True;
  685.    return cdc;
  686. }
  687. /*---------------------------------------------*/
  688. void compress ( Char *name )
  689. {
  690.    FILE *inStr;
  691.    FILE *outStr;
  692.    if (name == NULL && srcMode != SM_I2O)
  693.       panic ( "compress: bad modesn" );
  694.    switch (srcMode) {
  695.       case SM_I2O: copyFileName ( inName, "(stdin)" );
  696.                    copyFileName ( outName, "(stdout)" ); break;
  697.       case SM_F2F: copyFileName ( inName, name );
  698.                    copyFileName ( outName, name );
  699.                    strcat ( outName, ".bz2" ); break;
  700.       case SM_F2O: copyFileName ( inName, name );
  701.                    copyFileName ( outName, "(stdout)" ); break;
  702.    }
  703.    if ( srcMode != SM_I2O && containsDubiousChars ( inName ) ) {
  704.       fprintf ( stderr, "%s: There are no files matching `%s'.n",
  705.       progName, inName );
  706.       return;
  707.    }
  708.    if ( srcMode != SM_I2O && !fileExists ( inName ) ) {
  709.       fprintf ( stderr, "%s: Input file %s doesn't exist, skipping.n",
  710.                 progName, inName );
  711.       return;
  712.    }
  713.    if ( srcMode != SM_I2O && endsInBz2 ( inName )) {
  714.       fprintf ( stderr, "%s: Input file name %s ends in `.bz2', skipping.n",
  715.                 progName, inName );
  716.       return;
  717.    }
  718.    if ( srcMode != SM_I2O && notAStandardFile ( inName )) {
  719.       fprintf ( stderr, "%s: Input file %s is not a normal file, skipping.n",
  720.                 progName, inName );
  721.       return;
  722.    }
  723.    if ( srcMode == SM_F2F && !forceOverwrite && fileExists ( outName ) ) {
  724.       fprintf ( stderr, "%s: Output file %s already exists, skipping.n",
  725.                 progName, outName );
  726.       return;
  727.    }
  728.    switch ( srcMode ) {
  729.       case SM_I2O:
  730.          inStr = stdin;
  731.          outStr = stdout;
  732.          if ( isatty ( fileno ( stdout ) ) ) {
  733.             fprintf ( stderr,
  734.                       "%s: I won't write compressed data to a terminal.n",
  735.                       progName );
  736.             fprintf ( stderr, "%s: For help, type: `%s --help'.n",
  737.                               progName, progName );
  738.             return;
  739.          };
  740.          break;
  741.       case SM_F2O:
  742.          inStr = fopen ( inName, "rb" );
  743.          outStr = stdout;
  744.          if ( isatty ( fileno ( stdout ) ) ) {
  745.             fprintf ( stderr,
  746.                       "%s: I won't write compressed data to a terminal.n",
  747.                       progName );
  748.             fprintf ( stderr, "%s: For help, type: `%s --help'.n",
  749.                               progName, progName );
  750.             return;
  751.          };
  752.          if ( inStr == NULL ) {
  753.             fprintf ( stderr, "%s: Can't open input file %s, skipping.n",
  754.                       progName, inName );
  755.             return;
  756.          };
  757.          break;
  758.       case SM_F2F:
  759.          inStr = fopen ( inName, "rb" );
  760.          outStr = fopen ( outName, "wb" );
  761.          if ( outStr == NULL) {
  762.             fprintf ( stderr, "%s: Can't create output file %s, skipping.n",
  763.                       progName, outName );
  764.             return;
  765.          }
  766.          if ( inStr == NULL ) {
  767.             fprintf ( stderr, "%s: Can't open input file %s, skipping.n",
  768.                       progName, inName );
  769.             return;
  770.          };
  771.          setInterimPermissions ( outName );
  772.          break;
  773.       default:
  774.          panic ( "compress: bad srcMode" );
  775.          break;
  776.    }
  777.    if (verbosity >= 1) {
  778.       fprintf ( stderr,  "  %s: ", inName );
  779.       pad ( inName );
  780.       fflush ( stderr );
  781.    }
  782.    /*--- Now the input and output handles are sane.  Do the Biz. ---*/
  783.    outputHandleJustInCase = outStr;
  784.    compressStream ( inStr, outStr );
  785.    outputHandleJustInCase = NULL;
  786.    /*--- If there was an I/O error, we won't get here. ---*/
  787.    if ( srcMode == SM_F2F ) {
  788.       copyDatePermissionsAndOwner ( inName, outName );
  789.       if ( !keepInputFiles ) {
  790.          IntNative retVal = remove ( inName );
  791.          ERROR_IF_NOT_ZERO ( retVal );
  792.       }
  793.    }
  794. }
  795. /*---------------------------------------------*/
  796. void uncompress ( Char *name )
  797. {
  798.    FILE *inStr;
  799.    FILE *outStr;
  800.    Bool magicNumberOK;
  801.    if (name == NULL && srcMode != SM_I2O)
  802.       panic ( "uncompress: bad modesn" );
  803.    switch (srcMode) {
  804.       case SM_I2O: copyFileName ( inName, "(stdin)" );
  805.                    copyFileName ( outName, "(stdout)" ); break;
  806.       case SM_F2F: copyFileName ( inName, name );
  807.                    copyFileName ( outName, name );
  808.                    if (endsInBz2 ( outName ))
  809.                       outName [ strlen ( outName ) - 4 ] = '';
  810.                    break;
  811.       case SM_F2O: copyFileName ( inName, name );
  812.                    copyFileName ( outName, "(stdout)" ); break;
  813.    }
  814.    if ( srcMode != SM_I2O && containsDubiousChars ( inName ) ) {
  815.       fprintf ( stderr, "%s: There are no files matching `%s'.n",
  816.                 progName, inName );
  817.       return;
  818.    }
  819.    if ( srcMode != SM_I2O && !fileExists ( inName ) ) {
  820.       fprintf ( stderr, "%s: Input file %s doesn't exist, skipping.n",
  821.                 progName, inName );
  822.       return;
  823.    }
  824.    if ( srcMode != SM_I2O && !endsInBz2 ( inName )) {
  825.       fprintf ( stderr,
  826.                 "%s: Input file name %s doesn't end in `.bz2', skipping.n",
  827.                 progName, inName );
  828.       return;
  829.    }
  830.    if ( srcMode != SM_I2O && notAStandardFile ( inName )) {
  831.       fprintf ( stderr, "%s: Input file %s is not a normal file, skipping.n",
  832.                 progName, inName );
  833.       return;
  834.    }
  835.    if ( srcMode == SM_F2F && !forceOverwrite && fileExists ( outName ) ) {
  836.       fprintf ( stderr, "%s: Output file %s already exists, skipping.n",
  837.                 progName, outName );
  838.       return;
  839.    }
  840.    switch ( srcMode ) {
  841.       case SM_I2O:
  842.          inStr = stdin;
  843.          outStr = stdout;
  844.          if ( isatty ( fileno ( stdin ) ) ) {
  845.             fprintf ( stderr,
  846.                       "%s: I won't read compressed data from a terminal.n",
  847.                       progName );
  848.             fprintf ( stderr, "%s: For help, type: `%s --help'.n",
  849.                               progName, progName );
  850.             return;
  851.          };
  852.          break;
  853.       case SM_F2O:
  854.          inStr = fopen ( inName, "rb" );
  855.          outStr = stdout;
  856.          if ( inStr == NULL ) {
  857.             fprintf ( stderr, "%s: Can't open input file %s, skipping.n",
  858.                       progName, inName );
  859.             return;
  860.          };
  861.          break;
  862.       case SM_F2F:
  863.          inStr = fopen ( inName, "rb" );
  864.          outStr = fopen ( outName, "wb" );
  865.          if ( outStr == NULL) {
  866.             fprintf ( stderr, "%s: Can't create output file %s, skipping.n",
  867.                       progName, outName );
  868.             return;
  869.          }
  870.          if ( inStr == NULL ) {
  871.             fprintf ( stderr, "%s: Can't open input file %s, skipping.n",
  872.                       progName, inName );
  873.             return;
  874.          };
  875.          setInterimPermissions ( outName );
  876.          break;
  877.       default:
  878.          panic ( "uncompress: bad srcMode" );
  879.          break;
  880.    }
  881.    if (verbosity >= 1) {
  882.       fprintf ( stderr, "  %s: ", inName );
  883.       pad ( inName );
  884.       fflush ( stderr );
  885.    }
  886.    /*--- Now the input and output handles are sane.  Do the Biz. ---*/
  887.    outputHandleJustInCase = outStr;
  888.    magicNumberOK = uncompressStream ( inStr, outStr );
  889.    outputHandleJustInCase = NULL;
  890.    /*--- If there was an I/O error, we won't get here. ---*/
  891.    if ( magicNumberOK ) {
  892.       if ( srcMode == SM_F2F ) {
  893.          copyDatePermissionsAndOwner ( inName, outName );
  894.          if ( !keepInputFiles ) {
  895.             IntNative retVal = remove ( inName );
  896.             ERROR_IF_NOT_ZERO ( retVal );
  897.          }
  898.       }
  899.    } else {
  900.       if ( srcMode == SM_F2F ) {
  901.          IntNative retVal = remove ( outName );
  902.          ERROR_IF_NOT_ZERO ( retVal );
  903.       }
  904.    }
  905.    if ( magicNumberOK ) {
  906.       if (verbosity >= 1)
  907.          fprintf ( stderr, "donen" );
  908.    } else {
  909.       if (verbosity >= 1)
  910.          fprintf ( stderr, "not a bzip2 file, skipping.n" ); else
  911.          fprintf ( stderr,
  912.                    "%s: %s is not a bzip2 file, skipping.n",
  913.                    progName, inName );
  914.    }
  915. }
  916. /*---------------------------------------------*/
  917. void testf ( Char *name )
  918. {
  919.    FILE *inStr;
  920.    Bool allOK;
  921.    if (name == NULL && srcMode != SM_I2O)
  922.       panic ( "testf: bad modesn" );
  923.    copyFileName ( outName, "(none)" );
  924.    switch (srcMode) {
  925.       case SM_I2O: copyFileName ( inName, "(stdin)" ); break;
  926.       case SM_F2F: copyFileName ( inName, name ); break;
  927.       case SM_F2O: copyFileName ( inName, name ); break;
  928.    }
  929.    if ( srcMode != SM_I2O && containsDubiousChars ( inName ) ) {
  930.       fprintf ( stderr, "%s: There are no files matching `%s'.n",
  931.                 progName, inName );
  932.       return;
  933.    }
  934.    if ( srcMode != SM_I2O && !fileExists ( inName ) ) {
  935.       fprintf ( stderr, "%s: Input file %s doesn't exist, skipping.n",
  936.                 progName, inName );
  937.       return;
  938.    }
  939.    if ( srcMode != SM_I2O && !endsInBz2 ( inName )) {
  940.       fprintf ( stderr,
  941.                 "%s: Input file name %s doesn't end in `.bz2', skipping.n",
  942.                 progName, inName );
  943.       return;
  944.    }
  945.    if ( srcMode != SM_I2O && notAStandardFile ( inName )) {
  946.       fprintf ( stderr, "%s: Input file %s is not a normal file, skipping.n",
  947.                 progName, inName );
  948.       return;
  949.    }
  950.    switch ( srcMode ) {
  951.       case SM_I2O:
  952.          if ( isatty ( fileno ( stdin ) ) ) {
  953.             fprintf ( stderr,
  954.                       "%s: I won't read compressed data from a terminal.n",
  955.                       progName );
  956.             fprintf ( stderr, "%s: For help, type: `%s --help'.n",
  957.                               progName, progName );
  958.             return;
  959.          };
  960.          inStr = stdin;
  961.          break;
  962.       case SM_F2O: case SM_F2F:
  963.          inStr = fopen ( inName, "rb" );
  964.          if ( inStr == NULL ) {
  965.             fprintf ( stderr, "%s: Can't open input file %s, skipping.n",
  966.                       progName, inName );
  967.             return;
  968.          };
  969.          break;
  970.       default:
  971.          panic ( "testf: bad srcMode" );
  972.          break;
  973.    }
  974.    if (verbosity >= 1) {
  975.       fprintf ( stderr, "  %s: ", inName );
  976.       pad ( inName );
  977.       fflush ( stderr );
  978.    }
  979.    /*--- Now the input handle is sane.  Do the Biz. ---*/
  980.    allOK = testStream ( inStr );
  981.    if (allOK && verbosity >= 1) fprintf ( stderr, "okn" );
  982.    if (!allOK) testFailsExist = True;
  983. }
  984. /*---------------------------------------------*/
  985. void license ( void )
  986. {
  987.    fprintf ( stderr,
  988.     "bzip2, a block-sorting file compressor.  "
  989.     "Version 0.9.0c, 18-Oct-98.n"
  990.     "   n"
  991.     "   Copyright (C) 1996, 1997, 1998 by Julian Seward.n"
  992.     "   n"
  993.     "   This program is free software; you can redistribute it and/or modifyn"
  994.     "   it under the terms set out in the LICENSE file, which is includedn"
  995.     "   in the bzip2-0.9.0c source distribution.n"
  996.     "   n"
  997.     "   This program is distributed in the hope that it will be useful,n"
  998.     "   but WITHOUT ANY WARRANTY; without even the implied warranty ofn"
  999.     "   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See then"
  1000.     "   LICENSE file for more details.n"
  1001.     "   n"
  1002.    );
  1003. }
  1004. /*---------------------------------------------*/
  1005. void usage ( Char *fullProgName )
  1006. {
  1007.    fprintf (
  1008.       stderr,
  1009.       "bzip2, a block-sorting file compressor.  "
  1010.       "Version 0.9.0c, 18-Oct-98.n"
  1011.       "n   usage: %s [flags and input files in any order]n"
  1012.       "n"
  1013.       "   -h --help           print this messagen"
  1014.       "   -d --decompress     force decompressionn"
  1015.       "   -z --compress       force compressionn"
  1016.       "   -k --keep           keep (don't delete) input filesn"
  1017.       "   -f --force          overwrite existing output filessn"
  1018.       "   -t --test           test compressed file integrityn"
  1019.       "   -c --stdout         output to standard outn"
  1020.       "   -v --verbose        be verbose (a 2nd -v gives more)n"
  1021.       "   -L --license        display software version & licensen"
  1022.       "   -V --version        display software version & licensen"
  1023.       "   -s --small          use less memory (at most 2500k)n"
  1024.       "   -1 .. -9            set block size to 100k .. 900kn"
  1025.       "   --repetitive-fast   compress repetitive blocks fastern"
  1026.       "   --repetitive-best   compress repetitive blocks bettern"
  1027.       "n"
  1028.       "   If invoked as `bzip2', default action is to compress.n"
  1029.       "              as `bunzip2',  default action is to decompress.n"
  1030.       "              as `bz2cat', default action is to decompress to stdout.n"
  1031.       "n"
  1032.       "   If no file names are given, bzip2 compresses or decompressesn"
  1033.       "   from standard input to standard output.  You can combinen"
  1034.       "   short flags, so `-v -4' means the same as -v4 or -4v, &c.n"
  1035. #if BZ_UNIX
  1036.       "n"
  1037. #endif
  1038.       ,
  1039.       fullProgName
  1040.    );
  1041. }
  1042. /*---------------------------------------------*/
  1043. /*--
  1044.   All the garbage from here to main() is purely to
  1045.   implement a linked list of command-line arguments,
  1046.   into which main() copies argv[1 .. argc-1].
  1047.   The purpose of this ridiculous exercise is to
  1048.   facilitate the expansion of wildcard characters
  1049.   * and ? in filenames for halfwitted OSs like
  1050.   MSDOS, Windows 95 and NT.
  1051.   The actual Dirty Work is done by the platform-specific
  1052.   macro APPEND_FILESPEC.
  1053. --*/
  1054. typedef
  1055.    struct zzzz {
  1056.       Char        *name;
  1057.       struct zzzz *link;
  1058.    }
  1059.    Cell;
  1060. /*---------------------------------------------*/
  1061. void *myMalloc ( Int32 n )
  1062. {
  1063.    void* p;
  1064.    p = malloc ( (size_t)n );
  1065.    if (p == NULL) outOfMemory ();
  1066.    return p;
  1067. }
  1068. /*---------------------------------------------*/
  1069. Cell *mkCell ( void )
  1070. {
  1071.    Cell *c;
  1072.    c = (Cell*) myMalloc ( sizeof ( Cell ) );
  1073.    c->name = NULL;
  1074.    c->link = NULL;
  1075.    return c;
  1076. }
  1077. /*---------------------------------------------*/
  1078. Cell *snocString ( Cell *root, Char *name )
  1079. {
  1080.    if (root == NULL) {
  1081.       Cell *tmp = mkCell();
  1082.       tmp->name = (Char*) myMalloc ( 5 + strlen(name) );
  1083.       strcpy ( tmp->name, name );
  1084.       return tmp;
  1085.    } else {
  1086.       Cell *tmp = root;
  1087.       while (tmp->link != NULL) tmp = tmp->link;
  1088.       tmp->link = snocString ( tmp->link, name );
  1089.       return root;
  1090.    }
  1091. }
  1092. /*---------------------------------------------*/
  1093. #define ISFLAG(s) (strcmp(aa->name, (s))==0)
  1094. IntNative main ( IntNative argc, Char *argv[] )
  1095. {
  1096.    Int32  i, j;
  1097.    Char   *tmp;
  1098.    Cell   *argList;
  1099.    Cell   *aa;
  1100.    /*-- Be really really really paranoid :-) --*/
  1101.    if (sizeof(Int32) != 4 || sizeof(UInt32) != 4  ||
  1102.        sizeof(Int16) != 2 || sizeof(UInt16) != 2  ||
  1103.        sizeof(Char)  != 1 || sizeof(UChar)  != 1) {
  1104.       fprintf ( stderr,
  1105.                 "bzip2: I'm not configured correctly for this platform!n"
  1106.                 "tI require Int32, Int16 and Char to have sizesn"
  1107.                 "tof 4, 2 and 1 bytes to run properly, and they don't.n"
  1108.                 "tProbably you can fix this by defining them correctly,n"
  1109.                 "tand recompiling.  Bye!n" );
  1110.       exit(3);
  1111.    }
  1112.    /*-- Set up signal handlers --*/
  1113.    signal (SIGINT,  mySignalCatcher);
  1114.    signal (SIGTERM, mySignalCatcher);
  1115.    signal (SIGSEGV, mySIGSEGVorSIGBUScatcher);
  1116. #if BZ_UNIX
  1117.    signal (SIGHUP,  mySignalCatcher);
  1118.    signal (SIGBUS,  mySIGSEGVorSIGBUScatcher);
  1119. #endif
  1120.    /*-- Initialise --*/
  1121.    outputHandleJustInCase  = NULL;
  1122.    smallMode               = False;
  1123.    keepInputFiles          = False;
  1124.    forceOverwrite          = False;
  1125.    verbosity               = 0;
  1126.    blockSize100k           = 9;
  1127.    testFailsExist          = False;
  1128.    numFileNames            = 0;
  1129.    numFilesProcessed       = 0;
  1130.    workFactor              = 30;
  1131.    copyFileName ( inName,  "(none)" );
  1132.    copyFileName ( outName, "(none)" );
  1133.    copyFileName ( progNameReally, argv[0] );
  1134.    progName = &progNameReally[0];
  1135.    for (tmp = &progNameReally[0]; *tmp != ''; tmp++)
  1136.       if (*tmp == PATH_SEP) progName = tmp + 1;
  1137.    /*-- Expand filename wildcards in arg list --*/
  1138.    argList = NULL;
  1139.    for (i = 1; i <= argc-1; i++)
  1140.       APPEND_FILESPEC(argList, argv[i]);
  1141.    /*-- Find the length of the longest filename --*/
  1142.    longestFileName = 7;
  1143.    numFileNames    = 0;
  1144.    for (aa = argList; aa != NULL; aa = aa->link)
  1145.       if (aa->name[0] != '-') {
  1146.          numFileNames++;
  1147.          if (longestFileName < (Int32)strlen(aa->name) )
  1148.             longestFileName = (Int32)strlen(aa->name);
  1149.       }
  1150.    /*-- Determine source modes; flag handling may change this too. --*/
  1151.    if (numFileNames == 0)
  1152.       srcMode = SM_I2O; else srcMode = SM_F2F;
  1153.    /*-- Determine what to do (compress/uncompress/test/cat). --*/
  1154.    /*-- Note that subsequent flag handling may change this. --*/
  1155.    opMode = OM_Z;
  1156.    if ( (strstr ( progName, "unzip" ) != 0) ||
  1157.         (strstr ( progName, "UNZIP" ) != 0) )
  1158.       opMode = OM_UNZ;
  1159.    if ( (strstr ( progName, "z2cat" ) != 0) ||
  1160.         (strstr ( progName, "Z2CAT" ) != 0) ||
  1161.         (strstr ( progName, "zcat" ) != 0)  ||
  1162.         (strstr ( progName, "ZCAT" ) != 0) )  {
  1163.       opMode = OM_UNZ;
  1164.       srcMode = (numFileNames == 0) ? SM_I2O : SM_F2O;
  1165.    }
  1166.    /*-- Look at the flags. --*/
  1167.    for (aa = argList; aa != NULL; aa = aa->link)
  1168.       if (aa->name[0] == '-' && aa->name[1] != '-')
  1169.          for (j = 1; aa->name[j] != ''; j++)
  1170.             switch (aa->name[j]) {
  1171.                case 'c': srcMode          = SM_F2O; break;
  1172.                case 'd': opMode           = OM_UNZ; break;
  1173.                case 'z': opMode           = OM_Z; break;
  1174.                case 'f': forceOverwrite   = True; break;
  1175.                case 't': opMode           = OM_TEST; break;
  1176.                case 'k': keepInputFiles   = True; break;
  1177.                case 's': smallMode        = True; break;
  1178.                case '1': blockSize100k    = 1; break;
  1179.                case '2': blockSize100k    = 2; break;
  1180.                case '3': blockSize100k    = 3; break;
  1181.                case '4': blockSize100k    = 4; break;
  1182.                case '5': blockSize100k    = 5; break;
  1183.                case '6': blockSize100k    = 6; break;
  1184.                case '7': blockSize100k    = 7; break;
  1185.                case '8': blockSize100k    = 8; break;
  1186.                case '9': blockSize100k    = 9; break;
  1187.                case 'V':
  1188.                case 'L': license();            break;
  1189.                case 'v': verbosity++; break;
  1190.                case 'h': usage ( progName );
  1191.                          exit ( 1 );
  1192.                          break;
  1193.                default:  fprintf ( stderr, "%s: Bad flag `%s'n",
  1194.                                    progName, aa->name );
  1195.                          usage ( progName );
  1196.                          exit ( 1 );
  1197.                          break;
  1198.          }
  1199.    /*-- And again ... --*/
  1200.    for (aa = argList; aa != NULL; aa = aa->link) {
  1201.       if (ISFLAG("--stdout"))            srcMode          = SM_F2O;  else
  1202.       if (ISFLAG("--decompress"))        opMode           = OM_UNZ;  else
  1203.       if (ISFLAG("--compress"))          opMode           = OM_Z;    else
  1204.       if (ISFLAG("--force"))             forceOverwrite   = True;    else
  1205.       if (ISFLAG("--test"))              opMode           = OM_TEST; else
  1206.       if (ISFLAG("--keep"))              keepInputFiles   = True;    else
  1207.       if (ISFLAG("--small"))             smallMode        = True;    else
  1208.       if (ISFLAG("--version"))           license();                  else
  1209.       if (ISFLAG("--license"))           license();                  else
  1210.       if (ISFLAG("--repetitive-fast"))   workFactor = 5;             else
  1211.       if (ISFLAG("--repetitive-best"))   workFactor = 150;           else
  1212.       if (ISFLAG("--verbose"))           verbosity++;                else
  1213.       if (ISFLAG("--help"))              { usage ( progName ); exit ( 1 ); }
  1214.          else
  1215.          if (strncmp ( aa->name, "--", 2) == 0) {
  1216.             fprintf ( stderr, "%s: Bad flag `%s'n", progName, aa->name );
  1217.             usage ( progName );
  1218.             exit ( 1 );
  1219.          }
  1220.    }
  1221.    if (verbosity > 4) verbosity = 4;
  1222.    if (opMode == OM_Z && smallMode) blockSize100k = 2;
  1223.    if (srcMode == SM_F2O && numFileNames == 0) {
  1224.       fprintf ( stderr, "%s: -c expects at least one filename.n",
  1225.                 progName );
  1226.       exit ( 1 );
  1227.    }
  1228.    if (opMode == OM_TEST && srcMode == SM_F2O) {
  1229.       fprintf ( stderr, "%s: -c and -t cannot be used together.n",
  1230.                 progName );
  1231.       exit ( 1 );
  1232.    }
  1233.    if (opMode != OM_Z) blockSize100k = 0;
  1234.    if (opMode == OM_Z) {
  1235.       if (srcMode == SM_I2O)
  1236.          compress ( NULL );
  1237.          else
  1238.          for (aa = argList; aa != NULL; aa = aa->link)
  1239.             if (aa->name[0] != '-') {
  1240.                numFilesProcessed++;
  1241.                compress ( aa->name );
  1242.             }
  1243.    } else
  1244.    if (opMode == OM_UNZ) {
  1245.       if (srcMode == SM_I2O)
  1246.          uncompress ( NULL );
  1247.          else
  1248.          for (aa = argList; aa != NULL; aa = aa->link)
  1249.             if (aa->name[0] != '-') {
  1250.                numFilesProcessed++;
  1251.                uncompress ( aa->name );
  1252.             }
  1253.    } else {
  1254.       testFailsExist = False;
  1255.       if (srcMode == SM_I2O)
  1256.          testf ( NULL );
  1257.          else
  1258.          for (aa = argList; aa != NULL; aa = aa->link)
  1259.             if (aa->name[0] != '-') {
  1260.                numFilesProcessed++;
  1261.                testf ( aa->name );
  1262.             }
  1263.       if (testFailsExist) {
  1264.          fprintf ( stderr,
  1265.            "n"
  1266.            "You can use the `bzip2recover' program to *attempt* to recovern"
  1267.            "data from undamaged sections of corrupted files.nn"
  1268.          );
  1269.          exit(2);
  1270.       }
  1271.    }
  1272.    return 0;
  1273. }
  1274. /*-----------------------------------------------------------*/
  1275. /*--- end                                         bzip2.c ---*/
  1276. /*-----------------------------------------------------------*/