test-gsimm.con commit Merge branch 'jc/diffstat' into next (9849efb)
   1#include <unistd.h>
   2#include <stdlib.h>
   3#include <fcntl.h>
   4#include <libgen.h>
   5#include <stdio.h>
   6#include <assert.h>
   7#include <math.h>
   8#include <string.h>
   9#include <sys/types.h>
  10#include <sys/stat.h>
  11#include <sys/mman.h>
  12
  13#include "rabinpoly.h"
  14#include "gsimm.h"
  15
  16#define MIN(x,y) ((y)<(x) ? (y) : (x))
  17#define MAX(x,y) ((y)>(x) ? (y) : (x))
  18
  19/* The RABIN_WINDOW_SIZE is the size of fingerprint window used by
  20   Rabin algorithm. This is not a modifiable parameter.
  21
  22   The first RABIN_WINDOW_SIZE - 1 bytes are skipped, in order to ensure
  23   fingerprints are good hashes. This does somewhat reduce the
  24   influence of the first few bytes in the file (they're part of
  25   fewer windows, like the last few bytes), but that actually isn't
  26   so bad as files often start with fixed content that may bias comparisons.
  27*/
  28
  29typedef struct fileinfo
  30{ char          *name;
  31  size_t        length;
  32  u_char        md[MD_LENGTH];
  33  int           match;
  34} File;
  35
  36int flag_verbose = 0;
  37int flag_debug = 0;
  38char *flag_relative = 0;
  39
  40char cmd[12] = "        ...";
  41char md_strbuf[MD_LENGTH * 2 + 1];
  42u_char relative_md [MD_LENGTH];
  43
  44File *file;
  45int    file_count;
  46size_t file_bytes;
  47
  48char hex[17] = "0123456789abcdef";
  49
  50void usage()
  51{  fprintf (stderr, "usage: %s [-dhvw] [-r fingerprint] file ...\n", cmd);
  52   fprintf (stderr, " -d\tdebug output, repeate for more verbosity\n");
  53   fprintf (stderr, " -h\tshow this usage information\n");
  54   fprintf (stderr, " -r\tshow distance relative to fingerprint "
  55                    "(%u hex digits)\n", MD_LENGTH * 2);
  56   fprintf (stderr, " -v\tverbose output, repeat for even more verbosity\n");
  57   fprintf (stderr, " -w\tenable warnings for suspect statistics\n");
  58   exit (1);
  59}
  60
  61int dist (u_char *l, u_char *r)
  62{ int j, k;
  63  int d = 0;
  64
  65  for (j = 0; j < MD_LENGTH; j++)
  66  { u_char ch = l[j] ^ r[j];
  67
  68    for (k = 0; k < 8; k++) d += ((ch & (1<<k)) > 0);
  69  }
  70
  71  return d;
  72}
  73
  74char *md_to_str(u_char *md)
  75{ int j;
  76
  77  for (j = 0; j < MD_LENGTH; j++)
  78  { u_char ch = md[j];
  79
  80    md_strbuf[j*2] = hex[ch >> 4];
  81    md_strbuf[j*2+1] = hex[ch & 0xF];
  82  }
  83
  84  md_strbuf[j*2] = 0;
  85  return md_strbuf;
  86}
  87
  88void process_file (char *name)
  89{ int fd;
  90  struct stat fs;
  91  u_char *data;
  92  File *fi = file+file_count;;
  93
  94  fd = open (name, O_RDONLY, 0);
  95  if (fd < 0)
  96  { perror (name);
  97    exit (2);
  98  }
  99
 100  if (fstat (fd, &fs))
 101  { perror (name);
 102    exit (2);
 103  }
 104
 105  if (fs.st_size >= MIN_FILE_SIZE
 106      && fs.st_size <= MAX_FILE_SIZE)
 107  { fi->length = fs.st_size;
 108    fi->name = name;
 109
 110    data = (u_char *) mmap (0, fs.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
 111
 112    if (data == (u_char *) -1)
 113    { perror (name);
 114      exit (2);
 115    }
 116
 117    gb_simm_process (data, fs.st_size, fi->md);
 118    if (flag_relative)
 119    { int d = dist (fi->md, relative_md);
 120      double sim = 1.0 - MIN (1.0, (double) (d) / (MD_LENGTH * 4 - 1));
 121      fprintf (stdout, "%s %llu %u %s %u %3.1f\n",
 122               md_to_str (fi->md), (long long unsigned) 0,
 123               (unsigned) fs.st_size, name,
 124               d, 100.0 * sim);
 125    }
 126    else
 127    {
 128      fprintf (stdout, "%s %llu %u %s\n",
 129               md_to_str (fi->md), (long long unsigned) 0,
 130               (unsigned) fs.st_size, name);
 131    }
 132    munmap (data, fs.st_size);
 133    file_bytes += fs.st_size;
 134    file_count++;
 135  } else if (flag_verbose)
 136  { fprintf (stdout, "skipping %s (size %llu)\n", name, (long long unsigned) fs.st_size); }
 137
 138  close (fd);
 139}
 140
 141u_char *str_to_md(char *str, u_char *md)
 142{ int j;
 143
 144  if (!md || !str) return 0;
 145
 146  bzero (md, MD_LENGTH);
 147
 148  for (j = 0; j < MD_LENGTH * 2; j++)
 149  { char ch = str[j];
 150
 151    if (ch >= '0' && ch <= '9')
 152    { md [j/2] = (md [j/2] << 4) + (ch - '0');
 153    }
 154    else
 155    { ch |= 32;
 156
 157      if (ch < 'a' || ch > 'f') break;
 158      md [j/2] = (md[j/2] << 4) + (ch - 'a' + 10);
 159  } }
 160
 161  return (j != MD_LENGTH * 2 || str[j] != 0) ? 0 : md;
 162}
 163
 164int main (int argc, char *argv[])
 165{ int ch, j;
 166
 167  strncpy (cmd, basename (argv[0]), 8);
 168
 169  while ((ch = getopt(argc, argv, "dhr:vw")) != -1)
 170  { switch (ch)
 171    { case 'd': flag_debug++;
 172                break;
 173      case 'r': if (!optarg)
 174                { fprintf (stderr, "%s: missing argument for -r\n", cmd);
 175                  return 1;
 176                }
 177                if (str_to_md (optarg, relative_md)) flag_relative = optarg;
 178                else
 179                { fprintf (stderr, "%s: not a valid fingerprint\n", optarg);
 180                  return 1;
 181                }
 182                break;
 183      case 'v': flag_verbose++;
 184                break;
 185      case 'w': break;
 186      default : usage();
 187                return (ch != 'h');
 188  } }
 189
 190  argc -= optind;
 191  argv += optind;
 192
 193  if (argc == 0) usage();
 194
 195  rabin_reset ();
 196  if (flag_verbose && flag_relative)
 197  { fprintf (stdout, "distances are relative to %s\n", flag_relative);
 198  }
 199
 200  file = (File *) calloc (argc, sizeof (File));
 201
 202  for (j = 0; j < argc; j++) process_file (argv[j]);
 203
 204  if (flag_verbose)
 205  { fprintf (stdout, "%li bytes in %i files\n", (long) file_bytes, file_count);
 206  }
 207
 208  return 0;
 209}