1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287
|
/* code -- bigram- and front-encode filenames for locate
Copyright (C) 1994, 2005, 2007, 2008 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 <http://www.gnu.org/licenses/>.
*/
/* 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 <jwoods@adobe.com>.
Modified by David MacKenzie <djm@gnu.org>. */
#include <config.h>
#include <stdio.h>
#include <sys/types.h>
#include <string.h>
#include <errno.h>
#include <stdbool.h>
#ifdef STDC_HEADERS
#include <stdlib.h>
#endif
#if ENABLE_NLS
# include <libintl.h>
# 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
#include "locatedb.h"
#include "closeout.h"
#include "xalloc.h"
#include "gnulib-version.h"
#include "progname.h"
#include "error.h"
#include "findutils-version.h"
#ifndef ATTRIBUTE_NORETURN
# define ATTRIBUTE_NORETURN __attribute__ ((__noreturn__))
#endif
/* The name this program was run with. */
extern const char *program_name;
/* 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 <bug-findutils@gnu.org>.\n"), stream);
}
static void inerr (const char *filename) ATTRIBUTE_NORETURN;
static void outerr(void) ATTRIBUTE_NORETURN;
static void
inerr(const char *filename)
{
error(1, errno, "%s", filename);
/*NOTREACHED*/
abort();
}
static void
outerr(void)
{
error(1, 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]);
atexit (close_stdout);
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;
}
|