| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 | /* * Written by Daniel Richards <[email protected]> 6/7/2002 * hash.c: This app uses libtomcrypt to hash either stdin or a file * This file is Public Domain. No rights are reserved. * Compile with 'gcc hashsum.c -o hashsum -ltomcrypt' * This example isn't really big enough to warrent splitting into * more functions ;)*/#include <tomcrypt.h>#if _POSIX_C_SOURCE >= 200112L#include <libgen.h>#else#define basename(x) x#endif#include "common.h"#if !defined(PATH_MAX) && defined(_MSC_VER)#include <windows.h>#define PATH_MAX MAX_PATH#endif/* thanks http://stackoverflow.com/a/8198009 */#define _base(x) ((x >= '0' && x <= '9') ? '0' : \         (x >= 'a' && x <= 'f') ? 'a' - 10 : \         (x >= 'A' && x <= 'F') ? 'A' - 10 : \            '\255')#define HEXOF(x) (x - _base(x))static char* hashsum;static void die(int status){   unsigned long w, x;   FILE* o = status == EXIT_SUCCESS ? stdout : stderr;   fprintf(o, "usage: %s -a algorithm [-c] [file...]\n", hashsum);   fprintf(o, "Algorithms:\n");   w = 0;   for (x = 0; hash_descriptor[x].name != NULL; x++) {      w += fprintf(o, "%-14s", hash_descriptor[x].name);      if (w >= 70) {         fprintf(o, "\n");         w = 0;      }   }   if (w != 0) fprintf(o, "\n");   free(hashsum);   exit(status);}static void printf_hex(unsigned char* hash_buffer, unsigned long w){   unsigned long x;   for (x = 0; x < w; x++) {       printf("%02x",hash_buffer[x]);   }}static void check_file(int argn, int argc, char **argv){   int err, failed, invalid;   unsigned char is_buffer[MAXBLOCKSIZE], should_buffer[MAXBLOCKSIZE];   char buf[PATH_MAX + (MAXBLOCKSIZE * 3)];   /* iterate through all files */   while(argn < argc) {      char* s;      FILE* f = fopen(argv[argn], "rb");      if(f == NULL) {         int n = snprintf(buf, sizeof(buf), "%s: %s", hashsum, argv[argn]);         if (n > 0 && n < (int)sizeof(buf))            perror(buf);         else            perror(argv[argn]);         exit(EXIT_FAILURE);      }      failed = 0;      invalid = 0;      /* read the file line by line */      while((s = fgets(buf, sizeof(buf), f)) != NULL)      {         int tries, n;         unsigned long hash_len, w, x;         char* space = strstr(s, " ");         if (space == NULL) {            fprintf(stderr, "%s: no properly formatted checksum lines found\n", hashsum);            goto ERR;         }         hash_len = space - s;         hash_len /= 2;         /* convert the hex-string back to binary */         for (x = 0; x < hash_len; ++x) {            should_buffer[x] = HEXOF(s[x*2]) << 4 | HEXOF(s[x*2 + 1]);         }         space++;         if (*space != '*') {            fprintf(stderr, "%s: unsupported input mode '%c'\n", hashsum, *space);            goto ERR;         }         space++;         for (n = 0; n < (buf + sizeof(buf)) - space; ++n) {            if(iscntrl((int)space[n])) {               space[n] = '\0';               break;            }         }         /* try all hash algorithms that have the appropriate hash size */         tries = 0;         for (x = 0; hash_descriptor[x].name != NULL; ++x) {            if (hash_descriptor[x].hashsize == hash_len) {               tries++;               w = sizeof(is_buffer);               if ((err = hash_file(x, space, is_buffer, &w)) != CRYPT_OK) {                  fprintf(stderr, "%s: File hash error: %s: %s\n", hashsum, space, error_to_string(err));ERR:                  fclose(f);                  exit(EXIT_FAILURE);               }               if(XMEMCMP(should_buffer, is_buffer, w) == 0) {                  printf("%s: OK\n", space);                  break;               }            }         } /* for */         if (hash_descriptor[x].name == NULL) {            if(tries > 0) {               printf("%s: FAILED\n", space);               failed++;            }            else {               invalid++;            }         }      } /* while */      fclose(f);      if(invalid) {         fprintf(stderr, "%s: WARNING: %d %s is improperly formatted\n", hashsum, invalid, invalid > 1?"lines":"line");      }      if(failed) {         fprintf(stderr, "%s: WARNING: %d computed %s did NOT match\n", hashsum, failed, failed > 1?"checksums":"checksum");      }      argn++;   }   exit(EXIT_SUCCESS);}int main(int argc, char **argv){   int idx, check, z, err, argn;   unsigned long w, x;   unsigned char hash_buffer[MAXBLOCKSIZE];   hashsum = strdup(basename(argv[0]));   /* You need to register algorithms before using them */   register_algs();   if (argc > 1 && (strcmp("-h", argv[1]) == 0 || strcmp("--help", argv[1]) == 0)) {      die(EXIT_SUCCESS);   }   if (argc < 3) {      die(EXIT_FAILURE);   }   argn = 1;   check = 0;   idx = -2;   while(argn < argc){      if(strcmp("-a", argv[argn]) == 0) {         argn++;         if(argn < argc) {            idx = find_hash(argv[argn]);            if (idx == -1) {               struct {                  const char* is;                  const char* should;               } shasum_compat[] =                     {#ifdef LTC_SHA1                           { "1",        sha1_desc.name },#endif#ifdef LTC_SHA224                           { "224",      sha224_desc.name  },#endif#ifdef LTC_SHA256                           { "256",      sha256_desc.name  },#endif#ifdef LTC_SHA384                           { "384",      sha384_desc.name  },#endif#ifdef LTC_SHA512                           { "512",      sha512_desc.name  },#endif#ifdef LTC_SHA512_224                           { "512224",   sha512_224_desc.name  },#endif#ifdef LTC_SHA512_256                           { "512256",   sha512_256_desc.name  },#endif                           { NULL, NULL }                     };               for (x = 0; shasum_compat[x].is != NULL; ++x) {                  if(XSTRCMP(shasum_compat[x].is, argv[argn]) == 0) {                     idx = find_hash(shasum_compat[x].should);                     break;                  }               }            }            if (idx == -1) {               fprintf(stderr, "%s: Unrecognized algorithm\n", hashsum);               die(EXIT_FAILURE);            }            argn++;            continue;         }         else {            die(EXIT_FAILURE);         }      }      if(strcmp("-c", argv[argn]) == 0) {         check = 1;         argn++;         continue;      }      break;   }   if (check == 1) {      check_file(argn, argc, argv);   }   if (argc == argn) {      w = sizeof(hash_buffer);      if ((err = hash_filehandle(idx, stdin, hash_buffer, &w)) != CRYPT_OK) {         fprintf(stderr, "%s: File hash error: %s\n", hashsum, error_to_string(err));         return EXIT_FAILURE;      } else {          for (x = 0; x < w; x++) {              printf("%02x",hash_buffer[x]);          }          printf(" *-\n");      }   } else {      for (z = 3; z < argc; z++) {         w = sizeof(hash_buffer);         if ((err = hash_file(idx,argv[z],hash_buffer,&w)) != CRYPT_OK) {            fprintf(stderr, "%s: File hash error: %s\n", hashsum, error_to_string(err));            return EXIT_FAILURE;         } else {             printf_hex(hash_buffer, w);             printf(" *%s\n", argv[z]);         }      }   }   return EXIT_SUCCESS;}/* $Source$ *//* $Revision$ *//* $Date$ */
 |