home *** CD-ROM | disk | FTP | other *** search
/ The Developer Connection…ice Driver Kit for OS/2 3 / DEV3-D1.ISO / source / util2src / crcfile.c < prev    next >
Encoding:
C/C++ Source or Header  |  1992-06-07  |  15.5 KB  |  367 lines

  1. /*============================================================================*
  2.  * module: crcfile.c - Calculate the 16-bit CRC of a file.
  3.  *
  4.  * (C)Copyright IBM Corporation, 1990, 1991, 1992.        Brian E. Yoder
  5.  *
  6.  * This module contains the externally-available crcfile() subroutine.
  7.  * This subroutine opens a file, gets its length, calculates its
  8.  * CRC, and closes the file.
  9.  *
  10.  * The code to calculate the CRC and much of the its description was taken
  11.  * from the crc_chk.c module written by Jim Czenkusch in IBM Austin, TX.
  12.  * He did the hard part - I only took the code and wrapped another interface
  13.  * around it.
  14.  *
  15.  * 10/30/90 - Created from the crc_chk.c module written by Jim Czenkusch.
  16.  * 10/30/90 - Initial version, ported to the interface defined by crcfile().
  17.  * 05/02/91 - Ported from PS/2 AIX to DOS.  The file is explicitly opened
  18.  *            in binary mode.
  19.  * 06/07/92 - Added crcfile32() subroutine to calculate 32-bit CRCs.
  20.  *============================================================================*/
  21.  
  22. #include <stdio.h>
  23. #include <fcntl.h>
  24. #include <string.h>
  25. #include <memory.h>
  26. #include <sys/types.h>
  27. #include <sys/stat.h>
  28. #include <io.h>
  29.  
  30. #include "util.h"
  31.  
  32. #define NO       0
  33. #define YES      1
  34.  
  35. #define BUFFLEN  4096
  36. #define DEBUG    1
  37.  
  38. /*============================================================================*
  39.  * This table is directly from Jim's crc_chk.c, with no modifications other
  40.  * than making it 'static'.  It is used for calculating 16-bit CRCs.
  41.  *============================================================================*/
  42.  
  43. static unsigned short transition_tbl[256] =
  44.  
  45. {0x0000,0xA097,0xE1B9,0x412E,0x63E5,0xC372,0x825C,0x22CB,0xC7CA,0x675D,0x2673,
  46.  0x86E4,0xA42F,0x04B8,0x4596,0xE501,0x2F03,0x8F94,0xCEBA,0x6E2D,0x4CE6,0xEC71,
  47.  0xAD5F,0x0DC8,0xE8C9,0x485E,0x0970,0xA9E7,0x8B2C,0x2BBB,0x6A95,0xCA02,0x5E06,
  48.  0xFE91,0xBFBF,0x1F28,0x3DE3,0x9D74,0xDC5A,0x7CCD,0x99CC,0x395B,0x7875,0xD8E2,
  49.  0xFA29,0x5ABE,0x1B90,0xBB07,0x7105,0xD192,0x90BC,0x302B,0x12E0,0xB277,0xF359,
  50.  0x53CE,0xB6CF,0x1658,0x5776,0xF7E1,0xD52A,0x75BD,0x3493,0x9404,0xBC0C,0x1C9B,
  51.  0x5DB5,0xFD22,0xDFE9,0x7F7E,0x3E50,0x9EC7,0x7BC6,0xDB51,0x9A7F,0x3AE8,0x1823,
  52.  0xB8B4,0xF99A,0x590D,0x930F,0x3398,0x72B6,0xD221,0xF0EA,0x507D,0x1153,0xB1C4,
  53.  0x54C5,0xF452,0xB57C,0x15EB,0x3720,0x97B7,0xD699,0x760E,0xE20A,0x429D,0x03B3,
  54.  0xA324,0x81EF,0x2178,0x6056,0xC0C1,0x25C0,0x8557,0xC479,0x64EE,0x4625,0xE6B2,
  55.  0xA79C,0x070B,0xCD09,0x6D9E,0x2CB0,0x8C27,0xAEEC,0x0E7B,0x4F55,0xEFC2,0x0AC3,
  56.  0xAA54,0xEB7A,0x4BED,0x6926,0xC9B1,0x889F,0x2808,0xD88F,0x7818,0x3936,0x99A1,
  57.  0xBB6A,0x1BFD,0x5AD3,0xFA44,0x1F45,0xBFD2,0xFEFC,0x5E6B,0x7CA0,0xDC37,0x9D19,
  58.  0x3D8E,0xF78C,0x571B,0x1635,0xB6A2,0x9469,0x34FE,0x75D0,0xD547,0x3046,0x90D1,
  59.  0xD1FF,0x7168,0x53A3,0xF334,0xB21A,0x128D,0x8689,0x261E,0x6730,0xC7A7,0xE56C,
  60.  0x45FB,0x04D5,0xA442,0x4143,0xE1D4,0xA0FA,0x006D,0x22A6,0x8231,0xC31F,0x6388,
  61.  0xA98A,0x091D,0x4833,0xE8A4,0xCA6F,0x6AF8,0x2BD6,0x8B41,0x6E40,0xCED7,0x8FF9,
  62.  0x2F6E,0x0DA5,0xAD32,0xEC1C,0x4C8B,0x6483,0xC414,0x853A,0x25AD,0x0766,0xA7F1,
  63.  0xE6DF,0x4648,0xA349,0x03DE,0x42F0,0xE267,0xC0AC,0x603B,0x2115,0x8182,0x4B80,
  64.  0xEB17,0xAA39,0x0AAE,0x2865,0x88F2,0xC9DC,0x694B,0x8C4A,0x2CDD,0x6DF3,0xCD64,
  65.  0xEFAF,0x4F38,0x0E16,0xAE81,0x3A85,0x9A12,0xDB3C,0x7BAB,0x5960,0xF9F7,0xB8D9,
  66.  0x184E,0xFD4F,0x5DD8,0x1CF6,0xBC61,0x9EAA,0x3E3D,0x7F13,0xDF84,0x1586,0xB511,
  67.  0xF43F,0x54A8,0x7663,0xD6F4,0x97DA,0x374D,0xD24C,0x72DB,0x33F5,0x9362,0xB1A9,
  68.  0x113E,0x5010,0xF087};
  69.  
  70. /*===========================================================================*
  71.  * Lookup table for calculating 32-bit CRCs
  72.  *===========================================================================*/
  73.  
  74. static ulong Crc32Table[256] = {
  75.  
  76.           0x00000000 , 0x77073096 , 0xEE0E612C , 0x990951BA ,
  77.           0x076DC419 , 0x706AF48F , 0xE963A535 , 0x9E6495A3 ,
  78.           0x0EDB8832 , 0x79DCB8A4 , 0xE0D5E91E , 0x97D2D988 ,
  79.           0x09B64C2B , 0x7EB17CBD , 0xE7B82D07 , 0x90BF1D91 ,
  80.  
  81.           0x1DB71064 , 0x6AB020F2 , 0xF3B97148 , 0x84BE41DE ,
  82.           0x1ADAD47D , 0x6DDDE4EB , 0xF4D4B551 , 0x83D385C7 ,
  83.           0x136C9856 , 0x646BA8C0 , 0xFD62F97A , 0x8A65C9EC ,
  84.           0x14015C4F , 0x63066CD9 , 0xFA0F3D63 , 0x8D080DF5 ,
  85.  
  86.           0x3B6E20C8 , 0x4C69105E , 0xD56041E4 , 0xA2677172 ,
  87.           0x3C03E4D1 , 0x4B04D447 , 0xD20D85FD , 0xA50AB56B ,
  88.           0x35B5A8FA , 0x42B2986C , 0xDBBBC9D6 , 0xACBCF940 ,
  89.           0x32D86CE3 , 0x45DF5C75 , 0xDCD60DCF , 0xABD13D59 ,
  90.  
  91.           0x26D930AC , 0x51DE003A , 0xC8D75180 , 0xBFD06116 ,
  92.           0x21B4F4B5 , 0x56B3C423 , 0xCFBA9599 , 0xB8BDA50F ,
  93.           0x2802B89E , 0x5F058808 , 0xC60CD9B2 , 0xB10BE924 ,
  94.           0x2F6F7C87 , 0x58684C11 , 0xC1611DAB , 0xB6662D3D ,
  95.  
  96.           0x76DC4190 , 0x01DB7106 , 0x98D220BC , 0xEFD5102A ,
  97.           0x71B18589 , 0x06B6B51F , 0x9FBFE4A5 , 0xE8B8D433 ,
  98.           0x7807C9A2 , 0x0F00F934 , 0x9609A88E , 0xE10E9818 ,
  99.           0x7F6A0DBB , 0x086D3D2D , 0x91646C97 , 0xE6635C01 ,
  100.  
  101.           0x6B6B51F4 , 0x1C6C6162 , 0x856530D8 , 0xF262004E ,
  102.           0x6C0695ED , 0x1B01A57B , 0x8208F4C1 , 0xF50FC457 ,
  103.           0x65B0D9C6 , 0x12B7E950 , 0x8BBEB8EA , 0xFCB9887C ,
  104.           0x62DD1DDF , 0x15DA2D49 , 0x8CD37CF3 , 0xFBD44C65 ,
  105.  
  106.           0x4DB26158 , 0x3AB551CE , 0xA3BC0074 , 0xD4BB30E2 ,
  107.           0x4ADFA541 , 0x3DD895D7 , 0xA4D1C46D , 0xD3D6F4FB ,
  108.           0x4369E96A , 0x346ED9FC , 0xAD678846 , 0xDA60B8D0 ,
  109.           0x44042D73 , 0x33031DE5 , 0xAA0A4C5F , 0xDD0D7CC9 ,
  110.  
  111.           0x5005713C , 0x270241AA , 0xBE0B1010 , 0xC90C2086 ,
  112.           0x5768B525 , 0x206F85B3 , 0xB966D409 , 0xCE61E49F ,
  113.           0x5EDEF90E , 0x29D9C998 , 0xB0D09822 , 0xC7D7A8B4 ,
  114.           0x59B33D17 , 0x2EB40D81 , 0xB7BD5C3B , 0xC0BA6CAD ,
  115.  
  116.           0xEDB88320 , 0x9ABFB3B6 , 0x03B6E20C , 0x74B1D29A ,
  117.           0xEAD54739 , 0x9DD277AF , 0x04DB2615 , 0x73DC1683 ,
  118.           0xE3630B12 , 0x94643B84 , 0x0D6D6A3E , 0x7A6A5AA8 ,
  119.           0xE40ECF0B , 0x9309FF9D , 0x0A00AE27 , 0x7D079EB1 ,
  120.  
  121.           0xF00F9344 , 0x8708A3D2 , 0x1E01F268 , 0x6906C2FE ,
  122.           0xF762575D , 0x806567CB , 0x196C3671 , 0x6E6B06E7 ,
  123.           0xFED41B76 , 0x89D32BE0 , 0x10DA7A5A , 0x67DD4ACC ,
  124.           0xF9B9DF6F , 0x8EBEEFF9 , 0x17B7BE43 , 0x60B08ED5 ,
  125.  
  126.           0xD6D6A3E8 , 0xA1D1937E , 0x38D8C2C4 , 0x4FDFF252 ,
  127.           0xD1BB67F1 , 0xA6BC5767 , 0x3FB506DD , 0x48B2364B ,
  128.           0xD80D2BDA , 0xAF0A1B4C , 0x36034AF6 , 0x41047A60 ,
  129.           0xDF60EFC3 , 0xA867DF55 , 0x316E8EEF , 0x4669BE79 ,
  130.  
  131.           0xCB61B38C , 0xBC66831A , 0x256FD2A0 , 0x5268E236 ,
  132.           0xCC0C7795 , 0xBB0B4703 , 0x220216B9 , 0x5505262F ,
  133.           0xC5BA3BBE , 0xB2BD0B28 , 0x2BB45A92 , 0x5CB36A04 ,
  134.           0xC2D7FFA7 , 0xB5D0CF31 , 0x2CD99E8B , 0x5BDEAE1D ,
  135.  
  136.           0x9B64C2B0 , 0xEC63F226 , 0x756AA39C , 0x026D930A ,
  137.           0x9C0906A9 , 0xEB0E363F , 0x72076785 , 0x05005713 ,
  138.           0x95BF4A82 , 0xE2B87A14 , 0x7BB12BAE , 0x0CB61B38 ,
  139.           0x92D28E9B , 0xE5D5BE0D , 0x7CDCEFB7 , 0x0BDBDF21 ,
  140.  
  141.           0x86D3D2D4 , 0xF1D4E242 , 0x68DDB3F8 , 0x1FDA836E ,
  142.           0x81BE16CD , 0xF6B9265B , 0x6FB077E1 , 0x18B74777 ,
  143.           0x88085AE6 , 0xFF0F6A70 , 0x66063BCA , 0x11010B5C ,
  144.           0x8F659EFF , 0xF862AE69 , 0x616BFFD3 , 0x166CCF45 ,
  145.  
  146.           0xA00AE278 , 0xD70DD2EE , 0x4E048354 , 0x3903B3C2 ,
  147.           0xA7672661 , 0xD06016F7 , 0x4969474D , 0x3E6E77DB ,
  148.           0xAED16A4A , 0xD9D65ADC , 0x40DF0B66 , 0x37D83BF0 ,
  149.           0xA9BCAE53 , 0xDEBB9EC5 , 0x47B2CF7F , 0x30B5FFE9 ,
  150.  
  151.           0xBDBDF21C , 0xCABAC28A , 0x53B39330 , 0x24B4A3A6 ,
  152.           0xBAD03605 , 0xCDD70693 , 0x54DE5729 , 0x23D967BF ,
  153.           0xB3667A2E , 0xC4614AB8 , 0x5D681B02 , 0x2A6F2B94 ,
  154.           0xB40BBE37 , 0xC30C8EA1 , 0x5A05DF1B , 0x2D02EF8D
  155.  
  156. }; /* end of Crc32Table[] */
  157.  
  158. /*============================================================================*
  159.  * crcfile() - Gets a file's 16-bit CRC and length.
  160.  *
  161.  * PURPOSE:
  162.  *   This subroutine gets the length and CRC (cyclic redundancy check) value
  163.  *   associated with the file named 'fname'.
  164.  *
  165.  * RETURNS:
  166.  *   0, if successful.  The length is copied to *flen and the CRC is copied
  167.  *   to *crc.
  168.  *
  169.  *   If an error occurred (can't open or read the file), then 1 is returned.
  170.  *
  171.  * NOTE:
  172.  *   The code to calculate the CRC and much of the its description was taken
  173.  *   from the crc_chk.c module written by Jim Czenkusch in IBM Austin, TX.
  174.  *   He did the hard part - I only took the code and wrapped another interface
  175.  *   around it.
  176.  *
  177.  * REMARKS:
  178.  *   This program generates and validates the CRC signatures for a file.
  179.  *   The CRC signature is calculated using the following polynomial...
  180.  *
  181.  *     X**16 + X**15 + X**13 + X**7 + X**4 + X**2 + X + 1
  182.  *
  183.  *   where the register is initialized to 0x0000.  In addition to
  184.  *   calculating the file's CRC signature, the program also counts
  185.  *   the number of bytes that were processed to provided extra
  186.  *   insurance and allow some small measure of debug capacity.
  187.  *============================================================================*/
  188. int crcfile(fname, crc, flen)
  189.  
  190. char     *fname;                    /* Pointer to name of a message file */
  191. ushort   *crc;                      /* Pointer to location to store CRC in */
  192. ulong    *flen;                     /* Pointer to location to store length in */
  193.  
  194. {
  195.   int     rc;                       /* Return code storage */
  196.   int     cnt;                      /* Count */
  197.  
  198.   int     cfile;                    /* File handle */
  199.   ushort  crc_reg;                  /* CRC register value from the file */
  200.   ulong   actual_len;               /* Actual file length */
  201.  
  202.   unsigned char *data;              /* Pointer to data */
  203.   unsigned char in_buffer[BUFFLEN]; /* Buffer in which to read file */
  204.  
  205.   *crc = 0;                         /* Init. CRC value */
  206.   crc_reg = 0;
  207.  
  208.   *flen = 0L;                       /* Init. file length */
  209.   actual_len = 0L;
  210.  
  211.  /*---------------------------------------------------------------------------*
  212.   * Open the file for reading
  213.   *---------------------------------------------------------------------------*/
  214.  
  215.   cfile = open(fname, O_RDONLY | O_BINARY);
  216.   if (cfile == -1)
  217.      return(1);
  218.  
  219.  /*---------------------------------------------------------------------------*
  220.   * Calculate CRC for the file: Main Loop
  221.   *---------------------------------------------------------------------------*/
  222.  
  223.   for (;;)                          /* For each BUFFLEN block of the file: */
  224.   {
  225.     /*----------------------------------------------------------------------*
  226.      * Read a block of the file
  227.      *----------------------------------------------------------------------*/
  228.  
  229.      cnt = read(cfile, &in_buffer[0], BUFFLEN);
  230.      if (cnt == -1)                      /* If we got an error: */
  231.      {
  232.         close(cfile);                         /* Close the file */
  233.         return(1);                            /* Return with error */
  234.      }
  235.  
  236.      if (cnt == 0)                       /* If we read no data: */
  237.         break;                           /* We're done! */
  238.  
  239.     /*----------------------------------------------------------------------*
  240.      * Update the CRC value in 'crc_reg'
  241.      *----------------------------------------------------------------------*/
  242.  
  243.      data = in_buffer;
  244.      while (cnt > 0 )
  245.      {
  246.        crc_reg =
  247.           (crc_reg << 8) ^
  248.           transition_tbl[ ((crc_reg >> 8)^(unsigned short) (*(data++))) ];
  249.        actual_len++;
  250.        cnt--;
  251.      }
  252.  
  253.   } /* end of main for loop */
  254.  
  255.  /*---------------------------------------------------------------------------*
  256.   * Close the file, copy information back to caller, and return
  257.   *---------------------------------------------------------------------------*/
  258.  
  259.   close(cfile);
  260.  
  261.   *crc = crc_reg;
  262.   *flen = actual_len;
  263.  
  264.   return(0);
  265. }
  266.  
  267. /*============================================================================*
  268.  * crcfile32() - Gets a file's 32-bit CRC and length.
  269.  *
  270.  * PURPOSE:
  271.  *   This subroutine gets the length and CRC (cyclic redundancy check) value
  272.  *   associated with the file named 'fname'.
  273.  *
  274.  * RETURNS:
  275.  *   0, if successful.  The length is copied to *flen and the CRC is copied
  276.  *   to *crc.
  277.  *
  278.  *   If an error occurred (can't open or read the file), then 1 is returned.
  279.  *============================================================================*/
  280. int crcfile32(
  281.  
  282. char     *fname,                    /* Pointer to name of a message file */
  283. ulong    *crc,                      /* Pointer to location to store CRC in */
  284. ulong    *flen )                    /* Pointer to location to store length in */
  285.  
  286. {
  287.   int     rc;                       /* Return code storage */
  288.   int     cnt;                      /* Count */
  289.  
  290.   int     cfile;                    /* File handle */
  291.   ulong   crc_reg;                  /* CRC register value from the file */
  292.   ulong   actual_len;               /* Actual file length */
  293.  
  294.   ulong   a, b;                     /* Temp. ulong values */
  295.  
  296.   uchar  *data;                     /* Pointer to data */
  297.   uchar   buffer[BUFFLEN];          /* Buffer in which to read file */
  298.  
  299.   actual_len = 0L;                  /* Init. file length */
  300.   *flen = 0L;
  301.  
  302.  /*---------------------------------------------------------------------------*
  303.   * The CRC value (in crc_reg) is preconditioned with all 1s.  After the file
  304.   * has been processed, the CRC value is inverted (one's complement) to give
  305.   * us a value that is compatible with the value generated by PKZIP.
  306.   *---------------------------------------------------------------------------*/
  307.  
  308.   crc_reg = 0xFFFFFFFFL;            /* Init. CRC value */
  309.   *crc = crc_reg;
  310.  
  311.  /*---------------------------------------------------------------------------*
  312.   * Open the file for reading
  313.   *---------------------------------------------------------------------------*/
  314.  
  315.   cfile = open(fname, O_RDONLY | O_BINARY);
  316.   if (cfile == -1)
  317.      return(1);
  318.  
  319.  /*---------------------------------------------------------------------------*
  320.   * Calculate CRC for the file: Main Loop
  321.   *---------------------------------------------------------------------------*/
  322.  
  323.   for (;;)                          /* For each BUFFLEN block of the file: */
  324.   {
  325.     /*----------------------------------------------------------------------*
  326.      * Read a block of the file
  327.      *----------------------------------------------------------------------*/
  328.  
  329.      cnt = read(cfile, &buffer[0], BUFFLEN);
  330.      if (cnt == -1)                      /* If we got an error: */
  331.      {
  332.         close(cfile);                         /* Close the file */
  333.         return(1);                            /* Return with error */
  334.      }
  335.  
  336.      if (cnt == 0)                       /* If we read no data: */
  337.         break;                           /* We're done! */
  338.  
  339.     /*----------------------------------------------------------------------*
  340.      * Update the CRC value in 'crc_reg'
  341.      *----------------------------------------------------------------------*/
  342.  
  343.      data = buffer;
  344.      while (cnt > 0 )
  345.      {
  346.         a = (crc_reg >> 8) & 0x00FFFFFFL;
  347.         b = Crc32Table[ ( (int) crc_reg ^ *data++ ) & 0xFF ];
  348.         crc_reg = a ^ b;
  349.  
  350.         actual_len++;
  351.         cnt--;
  352.      }
  353.  
  354.   } /* end of main for loop */
  355.  
  356.  /*---------------------------------------------------------------------------*
  357.   * Close the file, copy information back to caller, and return
  358.   *---------------------------------------------------------------------------*/
  359.  
  360.   close(cfile);
  361.  
  362.   *crc = crc_reg ^ 0xFFFFFFFFL;
  363.   *flen = actual_len;
  364.  
  365.   return(0);
  366. }
  367.