/* code -- bigram- and front-encode filenames for locate Copyright (C) 1994, 2005, 2007, 2008, 2010, 2011 Free Software Foundation, Inc. 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 . */ /* Compress a sorted list. Works with `find' to encode a filename database to save space and search time. Usage: bigram < file_list > bigrams process-bigrams > most_common_bigrams code most_common_bigrams < file_list > squeezed_list Uses `front compression' (see ";login:", March 1983, p. 8). The output begins with the 128 most common bigrams. After that, the output format is, for each line, an offset (from the previous line) differential count byte followed by a (partially bigram-encoded) ASCII remainder. The output lines have no terminating byte; the start of the next line is indicated by its first byte having a value <= 30. The encoding of the output bytes is: 0-28 likeliest differential counts + offset (14) to make nonnegative 30 escape code for out-of-range count to follow in next halfword 128-255 bigram codes (the 128 most common, as determined by `updatedb') 32-127 single character (printable) ASCII remainder Written by James A. Woods . Modified by David MacKenzie . */ /* config.h should always be included first. */ #include /* system headers. */ #include #include #include #include #include #include /* gnulib headers. */ #include "closeout.h" #include "error.h" #include "gettext.h" #include "progname.h" #include "xalloc.h" /* find headers. */ #include "findutils-version.h" #include "locatedb.h" #if ENABLE_NLS # include # define _(Text) gettext (Text) #else # define _(Text) Text #define textdomain(Domain) #define bindtextdomain(Package, Directory) #endif #ifdef gettext_noop # define N_(String) gettext_noop (String) #else /* See locate.c for explanation as to why not use (String) */ # define N_(String) String #endif #ifndef ATTRIBUTE_NORETURN # define ATTRIBUTE_NORETURN __attribute__ ((__noreturn__)) #endif /* The 128 most common bigrams in the file list, padded with NULs if there are fewer. */ static char bigrams[257] = {0}; /* Return the offset of PATTERN in STRING, or -1 if not found. */ static int strindex (char *string, char *pattern) { register char *s; for (s = string; *s != '\0'; s++) /* Fast first char check. */ if (*s == *pattern) { register char *p2 = pattern + 1, *s2 = s + 1; while (*p2 != '\0' && *p2 == *s2) p2++, s2++; if (*p2 == '\0') return s2 - strlen (pattern) - string; } return -1; } /* Return the length of the longest common prefix of strings S1 and S2. */ static int prefix_length (char *s1, char *s2) { register char *start; for (start = s1; *s1 == *s2 && *s1 != '\0'; s1++, s2++) ; return s1 - start; } extern char *version_string; static void usage (FILE *stream) { fprintf (stream, _("\ Usage: %s [--version | --help]\n\ or %s most_common_bigrams < file-list > locate-database\n"), program_name, program_name); fputs (_("\nReport bugs to .\n"), stream); } static void inerr (const char *filename) ATTRIBUTE_NORETURN; static void outerr (void) ATTRIBUTE_NORETURN; static void inerr (const char *filename) { error (EXIT_FAILURE, errno, "%s", filename); /*NOTREACHED*/ abort (); } static void outerr (void) { error (EXIT_FAILURE, errno, _("write error")); /*NOTREACHED*/ abort (); } int main (int argc, char **argv) { char *path; /* The current input entry. */ char *oldpath; /* The previous input entry. */ size_t pathsize, oldpathsize; /* Amounts allocated for them. */ int count, oldcount, diffcount; /* Their prefix lengths & the difference. */ char bigram[3]; /* Bigram to search for in table. */ int code; /* Index of `bigram' in bigrams table. */ FILE *fp; /* Most common bigrams file. */ int line_len; /* Length of input line. */ set_program_name (argv[0]); if (atexit (close_stdout)) { error (EXIT_FAILURE, errno, _("The atexit library function failed")); } bigram[2] = '\0'; if (argc != 2) { usage (stderr); return 2; } if (0 == strcmp (argv[1], "--help")) { usage (stdout); return 0; } else if (0 == strcmp (argv[1], "--version")) { display_findutils_version ("code"); return 0; } fp = fopen (argv[1], "r"); if (fp == NULL) { fprintf (stderr, "%s: ", argv[0]); perror (argv[1]); return 1; } pathsize = oldpathsize = 1026; /* Increased as necessary by getline. */ path = xmalloc (pathsize); oldpath = xmalloc (oldpathsize); /* Set to empty string, to force the first prefix count to 0. */ oldpath[0] = '\0'; oldcount = 0; /* Copy the list of most common bigrams to the output, padding with NULs if there are <128 of them. */ if (NULL == fgets (bigrams, 257, fp)) inerr (argv[1]); if (256 != fwrite (bigrams, 1, 256, stdout)) outerr (); if (EOF == fclose (fp)) inerr (argv[1]); while ((line_len = getline (&path, &pathsize, stdin)) > 0) { char *pp; path[line_len - 1] = '\0'; /* Remove newline. */ /* Squelch unprintable chars in path so as not to botch decoding. */ for (pp = path; *pp != '\0'; pp++) { if (!(*pp >= 040 && *pp < 0177)) *pp = '?'; } count = prefix_length (oldpath, path); diffcount = count - oldcount; oldcount = count; /* If the difference is small, it fits in one byte; otherwise, two bytes plus a marker noting that fact. */ if (diffcount < -LOCATEDB_OLD_OFFSET || diffcount > LOCATEDB_OLD_OFFSET) { if (EOF ==- putc (LOCATEDB_OLD_ESCAPE, stdout)) outerr (); if (!putword (stdout, diffcount+LOCATEDB_OLD_OFFSET, GetwordEndianStateNative)) outerr (); } else { if (EOF == putc (diffcount + LOCATEDB_OLD_OFFSET, stdout)) outerr (); } /* Look for bigrams in the remainder of the path. */ for (pp = path + count; *pp != '\0'; pp += 2) { if (pp[1] == '\0') { /* No bigram is possible; only one char is left. */ putchar (*pp); break; } bigram[0] = *pp; bigram[1] = pp[1]; /* Linear search for specific bigram in string table. */ code = strindex (bigrams, bigram); if (code % 2 == 0) putchar ((code / 2) | 0200); /* It's a common bigram. */ else fputs (bigram, stdout); /* Write the text as printable ASCII. */ } { /* Swap path and oldpath and their sizes. */ char *tmppath = oldpath; size_t tmppathsize = oldpathsize; oldpath = path; oldpathsize = pathsize; path = tmppath; pathsize = tmppathsize; } } free (path); free (oldpath); return 0; }