// // cook - file construction tool // Copyright (C) 1994, 2003-2006, 2008, 2011, 2012 Peter Miller. // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 3 of the License, or (at // your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // // Derived from code marked: // Daniel J. Bernstein, brnstnd@nyu.edu. // 930708: fprintfile 0.95. Public domain. // 930708: Added fingerprintfile_addn. // 930622: Split off fprintfmt.c. // 930601: Baseline, fprintfile 0.8. Public domain. // No known patent problems. // #include #include #include #include #include #include #include struct combined_ty { FINGERPRINT_BASE_CLASS fingerprint_ty *snefru; fingerprint_ty *md5; fingerprint_ty *crc32; fingerprint_ty *len; }; static void combined_constructor(fingerprint_ty *p) { combined_ty *f; f = (combined_ty *)p; f->snefru = fingerprint_new(&fp_snefru); f->md5 = fingerprint_new(&fp_md5); f->crc32 = fingerprint_new(&fp_crc32); f->len = fingerprint_new(&fp_len); } static void combined_destructor(fingerprint_ty *p) { combined_ty *f; f = (combined_ty *)p; fingerprint_delete(f->snefru); fingerprint_delete(f->md5); fingerprint_delete(f->crc32); fingerprint_delete(f->len); } static void combined_addn(fingerprint_ty *p, unsigned char *s, size_t n) { combined_ty *f; f = (combined_ty *)p; fingerprint_addn(f->snefru, s, n); fingerprint_addn(f->md5, s, n); fingerprint_addn(f->crc32, s, n); fingerprint_addn(f->len, s, n); } static int combined_hash(fingerprint_ty *p, unsigned char *h) { combined_ty *f; int nbytes; unsigned char *obuf; f = (combined_ty *)p; obuf = h; nbytes = fingerprint_hash(f->snefru, h); h += nbytes; nbytes = fingerprint_hash(f->md5, h); h += nbytes; nbytes = fingerprint_hash(f->crc32, h); h += nbytes; nbytes = fingerprint_hash(f->len, h); h += nbytes; return (h - obuf); } static char base64sane[] = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ:."; static void combined_sum(fingerprint_ty *p, char *s, size_t data_len) { unsigned char h[1024]; int i; unsigned long x; #ifdef DEBUG int nbytes; nbytes = combined_hash(p, h); assert(nbytes == 57); #else combined_hash(p, h); #endif for (i = 0; i < 19; ++i) { unsigned idx; x = (h[3 * i] + 256L * (h[3 * i + 1] + 256L * h[3 * i + 2])); idx = (12 * i) % 76; if (idx < data_len) s[idx] = base64sane[x & 63]; x /= 64; idx = (12 * i + 41) % 76; if (idx < data_len) s[idx] = base64sane[x & 63]; x /= 64; idx = (12 * i + 6) % 76; if (idx < data_len) s[idx] = base64sane[x & 63]; x /= 64; idx = (12 * i + 47) % 76; if (idx < data_len) s[idx] = base64sane[x & 63]; } if (data_len < 77) s[data_len - 1] = 0; else s[76] = 0; } fingerprint_methods_ty fp_combined = { sizeof(combined_ty), "fingerprint", combined_constructor, combined_destructor, combined_addn, combined_hash, combined_sum }; // vim: set ts=8 sw=4 et :