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}