doc: refactor and update expand and unexpand --help
[coreutils.git] / src / ptx.c
blobc0c9733a2b17398c55b7c6ead1d88b00740bb9b0
1 /* Permuted index for GNU, with keywords in their context.
2 Copyright (C) 1990-2017 Free Software Foundation, Inc.
3 François Pinard <pinard@iro.umontreal.ca>, 1988.
5 This program is free software: you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation, either version 3 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
18 François Pinard <pinard@iro.umontreal.ca> */
20 #include <config.h>
22 #include <getopt.h>
23 #include <sys/types.h>
24 #include "system.h"
25 #include "die.h"
26 #include <regex.h>
27 #include "argmatch.h"
28 #include "diacrit.h"
29 #include "error.h"
30 #include "fadvise.h"
31 #include "quote.h"
32 #include "read-file.h"
33 #include "stdio--.h"
34 #include "xstrtol.h"
36 /* The official name of this program (e.g., no 'g' prefix). */
37 #define PROGRAM_NAME "ptx"
39 /* TRANSLATORS: Please translate "F. Pinard" to "François Pinard"
40 if "ç" (c-with-cedilla) is available in the translation's character
41 set and encoding. */
42 #define AUTHORS proper_name_utf8 ("F. Pinard", "Fran\xc3\xa7ois Pinard")
44 /* Number of possible characters in a byte. */
45 #define CHAR_SET_SIZE 256
47 #define ISODIGIT(C) ((C) >= '0' && (C) <= '7')
48 #define HEXTOBIN(C) ((C) >= 'a' && (C) <= 'f' ? (C)-'a'+10 \
49 : (C) >= 'A' && (C) <= 'F' ? (C)-'A'+10 : (C)-'0')
50 #define OCTTOBIN(C) ((C) - '0')
52 /* Debugging the memory allocator. */
54 #if WITH_DMALLOC
55 # define MALLOC_FUNC_CHECK 1
56 # include <dmalloc.h>
57 #endif
59 /* Global definitions. */
61 /* FIXME: There are many unchecked integer overflows in this file,
62 that will cause this command to misbehave given large inputs or
63 options. Many of the "int" values below should be "size_t" or
64 something else like that. */
66 /* Program options. */
68 enum Format
70 UNKNOWN_FORMAT, /* output format still unknown */
71 DUMB_FORMAT, /* output for a dumb terminal */
72 ROFF_FORMAT, /* output for 'troff' or 'nroff' */
73 TEX_FORMAT /* output for 'TeX' or 'LaTeX' */
76 static bool gnu_extensions = true; /* trigger all GNU extensions */
77 static bool auto_reference = false; /* refs are 'file_name:line_number:' */
78 static bool input_reference = false; /* refs at beginning of input lines */
79 static bool right_reference = false; /* output refs after right context */
80 static int line_width = 72; /* output line width in characters */
81 static int gap_size = 3; /* number of spaces between output fields */
82 static const char *truncation_string = "/";
83 /* string used to mark line truncations */
84 static const char *macro_name = "xx"; /* macro name for roff or TeX output */
85 static enum Format output_format = UNKNOWN_FORMAT;
86 /* output format */
88 static bool ignore_case = false; /* fold lower to upper for sorting */
89 static const char *break_file = NULL; /* name of the 'Break chars' file */
90 static const char *only_file = NULL; /* name of the 'Only words' file */
91 static const char *ignore_file = NULL; /* name of the 'Ignore words' file */
93 /* Options that use regular expressions. */
94 struct regex_data
96 /* The original regular expression, as a string. */
97 char const *string;
99 /* The compiled regular expression, and its fastmap. */
100 struct re_pattern_buffer pattern;
101 char fastmap[UCHAR_MAX + 1];
104 static struct regex_data context_regex; /* end of context */
105 static struct regex_data word_regex; /* keyword */
107 /* A BLOCK delimit a region in memory of arbitrary size, like the copy of a
108 whole file. A WORD is something smaller, its length should fit in a
109 short integer. A WORD_TABLE may contain several WORDs. */
111 typedef struct
113 char *start; /* pointer to beginning of region */
114 char *end; /* pointer to end + 1 of region */
116 BLOCK;
118 typedef struct
120 char *start; /* pointer to beginning of region */
121 short int size; /* length of the region */
123 WORD;
125 typedef struct
127 WORD *start; /* array of WORDs */
128 size_t alloc; /* allocated length */
129 size_t length; /* number of used entries */
131 WORD_TABLE;
133 /* Pattern description tables. */
135 /* For each character, provide its folded equivalent. */
136 static unsigned char folded_chars[CHAR_SET_SIZE];
138 /* End of context pattern register indices. */
139 static struct re_registers context_regs;
141 /* Keyword pattern register indices. */
142 static struct re_registers word_regs;
144 /* A word characters fastmap is used only when no word regexp has been
145 provided. A word is then made up of a sequence of one or more characters
146 allowed by the fastmap. Contains !0 if character allowed in word. Not
147 only this is faster in most cases, but it simplifies the implementation
148 of the Break files. */
149 static char word_fastmap[CHAR_SET_SIZE];
151 /* Maximum length of any word read. */
152 static int maximum_word_length;
154 /* Maximum width of any reference used. */
155 static int reference_max_width;
157 /* Ignore and Only word tables. */
159 static WORD_TABLE ignore_table; /* table of words to ignore */
160 static WORD_TABLE only_table; /* table of words to select */
162 /* Source text table, and scanning macros. */
164 static int number_input_files; /* number of text input files */
165 static int total_line_count; /* total number of lines seen so far */
166 static const char **input_file_name; /* array of text input file names */
167 static int *file_line_count; /* array of 'total_line_count' values at end */
169 static BLOCK *text_buffers; /* files to study */
171 /* SKIP_NON_WHITE used only for getting or skipping the reference. */
173 #define SKIP_NON_WHITE(cursor, limit) \
174 while (cursor < limit && ! isspace (to_uchar (*cursor))) \
175 cursor++
177 #define SKIP_WHITE(cursor, limit) \
178 while (cursor < limit && isspace (to_uchar (*cursor))) \
179 cursor++
181 #define SKIP_WHITE_BACKWARDS(cursor, start) \
182 while (cursor > start && isspace (to_uchar (cursor[-1]))) \
183 cursor--
185 #define SKIP_SOMETHING(cursor, limit) \
186 if (word_regex.string) \
188 regoff_t count; \
189 count = re_match (&word_regex.pattern, cursor, limit - cursor, 0, NULL); \
190 if (count == -2) \
191 matcher_error (); \
192 cursor += count == -1 ? 1 : count; \
194 else if (word_fastmap[to_uchar (*cursor)]) \
195 while (cursor < limit && word_fastmap[to_uchar (*cursor)]) \
196 cursor++; \
197 else \
198 cursor++
200 /* Occurrences table.
202 The 'keyword' pointer provides the central word, which is surrounded
203 by a left context and a right context. The 'keyword' and 'length'
204 field allow full 8-bit characters keys, even including NULs. At other
205 places in this program, the name 'keyafter' refers to the keyword
206 followed by its right context.
208 The left context does not extend, towards the beginning of the file,
209 further than a distance given by the 'left' value. This value is
210 relative to the keyword beginning, it is usually negative. This
211 insures that, except for white space, we will never have to backward
212 scan the source text, when it is time to generate the final output
213 lines.
215 The right context, indirectly attainable through the keyword end, does
216 not extend, towards the end of the file, further than a distance given
217 by the 'right' value. This value is relative to the keyword
218 beginning, it is usually positive.
220 When automatic references are used, the 'reference' value is the
221 overall line number in all input files read so far, in this case, it
222 is of type (int). When input references are used, the 'reference'
223 value indicates the distance between the keyword beginning and the
224 start of the reference field, it is of type (DELTA) and usually
225 negative. */
227 typedef short int DELTA; /* to hold displacement within one context */
229 typedef struct
231 WORD key; /* description of the keyword */
232 DELTA left; /* distance to left context start */
233 DELTA right; /* distance to right context end */
234 int reference; /* reference descriptor */
235 size_t file_index; /* corresponding file */
237 OCCURS;
239 /* The various OCCURS tables are indexed by the language. But the time
240 being, there is no such multiple language support. */
242 static OCCURS *occurs_table[1]; /* all words retained from the read text */
243 static size_t occurs_alloc[1]; /* allocated size of occurs_table */
244 static size_t number_of_occurs[1]; /* number of used slots in occurs_table */
247 /* Communication among output routines. */
249 /* Indicate if special output processing is requested for each character. */
250 static char edited_flag[CHAR_SET_SIZE];
252 static int half_line_width; /* half of line width, reference excluded */
253 static int before_max_width; /* maximum width of before field */
254 static int keyafter_max_width; /* maximum width of keyword-and-after field */
255 static int truncation_string_length;/* length of string that flags truncation */
257 /* When context is limited by lines, wraparound may happen on final output:
258 the 'head' pointer gives access to some supplementary left context which
259 will be seen at the end of the output line, the 'tail' pointer gives
260 access to some supplementary right context which will be seen at the
261 beginning of the output line. */
263 static BLOCK tail; /* tail field */
264 static int tail_truncation; /* flag truncation after the tail field */
266 static BLOCK before; /* before field */
267 static int before_truncation; /* flag truncation before the before field */
269 static BLOCK keyafter; /* keyword-and-after field */
270 static int keyafter_truncation; /* flag truncation after the keyafter field */
272 static BLOCK head; /* head field */
273 static int head_truncation; /* flag truncation before the head field */
275 static BLOCK reference; /* reference field for input reference mode */
277 /* Miscellaneous routines. */
279 /* Diagnose an error in the regular expression matcher. Then exit. */
281 static void ATTRIBUTE_NORETURN
282 matcher_error (void)
284 die (EXIT_FAILURE, errno, _("error in regular expression matcher"));
287 /*------------------------------------------------------.
288 | Duplicate string STRING, while evaluating \-escapes. |
289 `------------------------------------------------------*/
291 /* Loosely adapted from GNU sh-utils printf.c code. */
293 static char *
294 copy_unescaped_string (const char *string)
296 char *result; /* allocated result */
297 char *cursor; /* cursor in result */
298 int value; /* value of \nnn escape */
299 int length; /* length of \nnn escape */
301 result = xmalloc (strlen (string) + 1);
302 cursor = result;
304 while (*string)
306 if (*string == '\\')
308 string++;
309 switch (*string)
311 case 'x': /* \xhhh escape, 3 chars maximum */
312 value = 0;
313 for (length = 0, string++;
314 length < 3 && isxdigit (to_uchar (*string));
315 length++, string++)
316 value = value * 16 + HEXTOBIN (*string);
317 if (length == 0)
319 *cursor++ = '\\';
320 *cursor++ = 'x';
322 else
323 *cursor++ = value;
324 break;
326 case '0': /* \0ooo escape, 3 chars maximum */
327 value = 0;
328 for (length = 0, string++;
329 length < 3 && ISODIGIT (*string);
330 length++, string++)
331 value = value * 8 + OCTTOBIN (*string);
332 *cursor++ = value;
333 break;
335 case 'a': /* alert */
336 #if __STDC__
337 *cursor++ = '\a';
338 #else
339 *cursor++ = 7;
340 #endif
341 string++;
342 break;
344 case 'b': /* backspace */
345 *cursor++ = '\b';
346 string++;
347 break;
349 case 'c': /* cancel the rest of the output */
350 while (*string)
351 string++;
352 break;
354 case 'f': /* form feed */
355 *cursor++ = '\f';
356 string++;
357 break;
359 case 'n': /* new line */
360 *cursor++ = '\n';
361 string++;
362 break;
364 case 'r': /* carriage return */
365 *cursor++ = '\r';
366 string++;
367 break;
369 case 't': /* horizontal tab */
370 *cursor++ = '\t';
371 string++;
372 break;
374 case 'v': /* vertical tab */
375 #if __STDC__
376 *cursor++ = '\v';
377 #else
378 *cursor++ = 11;
379 #endif
380 string++;
381 break;
383 case '\0': /* lone backslash at end of string */
384 /* ignore it */
385 break;
387 default:
388 *cursor++ = '\\';
389 *cursor++ = *string++;
390 break;
393 else
394 *cursor++ = *string++;
397 *cursor = '\0';
398 return result;
401 /*--------------------------------------------------------------------------.
402 | Compile the regex represented by REGEX, diagnose and abort if any error. |
403 `--------------------------------------------------------------------------*/
405 static void
406 compile_regex (struct regex_data *regex)
408 struct re_pattern_buffer *pattern = &regex->pattern;
409 char const *string = regex->string;
410 char const *message;
412 pattern->buffer = NULL;
413 pattern->allocated = 0;
414 pattern->fastmap = regex->fastmap;
415 pattern->translate = ignore_case ? folded_chars : NULL;
417 message = re_compile_pattern (string, strlen (string), pattern);
418 if (message)
419 die (EXIT_FAILURE, 0, _("%s (for regexp %s)"), message, quote (string));
421 /* The fastmap should be compiled before 're_match'. The following
422 call is not mandatory, because 're_search' is always called sooner,
423 and it compiles the fastmap if this has not been done yet. */
425 re_compile_fastmap (pattern);
428 /*------------------------------------------------------------------------.
429 | This will initialize various tables for pattern match and compiles some |
430 | regexps. |
431 `------------------------------------------------------------------------*/
433 static void
434 initialize_regex (void)
436 int character; /* character value */
438 /* Initialize the case folding table. */
440 if (ignore_case)
441 for (character = 0; character < CHAR_SET_SIZE; character++)
442 folded_chars[character] = toupper (character);
444 /* Unless the user already provided a description of the end of line or
445 end of sentence sequence, select an end of line sequence to compile.
446 If the user provided an empty definition, thus disabling end of line
447 or sentence feature, make it NULL to speed up tests. If GNU
448 extensions are enabled, use end of sentence like in GNU emacs. If
449 disabled, use end of lines. */
451 if (context_regex.string)
453 if (!*context_regex.string)
454 context_regex.string = NULL;
456 else if (gnu_extensions && !input_reference)
457 context_regex.string = "[.?!][]\"')}]*\\($\\|\t\\| \\)[ \t\n]*";
458 else
459 context_regex.string = "\n";
461 if (context_regex.string)
462 compile_regex (&context_regex);
464 /* If the user has already provided a non-empty regexp to describe
465 words, compile it. Else, unless this has already been done through
466 a user provided Break character file, construct a fastmap of
467 characters that may appear in a word. If GNU extensions enabled,
468 include only letters of the underlying character set. If disabled,
469 include almost everything, even punctuations; stop only on white
470 space. */
472 if (word_regex.string)
473 compile_regex (&word_regex);
474 else if (!break_file)
476 if (gnu_extensions)
479 /* Simulate \w+. */
481 for (character = 0; character < CHAR_SET_SIZE; character++)
482 word_fastmap[character] = !! isalpha (character);
484 else
487 /* Simulate [^ \t\n]+. */
489 memset (word_fastmap, 1, CHAR_SET_SIZE);
490 word_fastmap[' '] = 0;
491 word_fastmap['\t'] = 0;
492 word_fastmap['\n'] = 0;
497 /*------------------------------------------------------------------------.
498 | This routine will attempt to swallow a whole file name FILE_NAME into a |
499 | contiguous region of memory and return a description of it into BLOCK. |
500 | Standard input is assumed whenever FILE_NAME is NULL, empty or "-". |
502 | Previously, in some cases, white space compression was attempted while |
503 | inputting text. This was defeating some regexps like default end of |
504 | sentence, which checks for two consecutive spaces. If white space |
505 | compression is ever reinstated, it should be in output routines. |
506 `------------------------------------------------------------------------*/
508 static void
509 swallow_file_in_memory (const char *file_name, BLOCK *block)
511 size_t used_length; /* used length in memory buffer */
513 /* As special cases, a file name which is NULL or "-" indicates standard
514 input, which is already opened. In all other cases, open the file from
515 its name. */
516 bool using_stdin = !file_name || !*file_name || STREQ (file_name, "-");
517 if (using_stdin)
518 block->start = fread_file (stdin, &used_length);
519 else
520 block->start = read_file (file_name, &used_length);
522 if (!block->start)
523 die (EXIT_FAILURE, errno, "%s", quotef (using_stdin ? "-" : file_name));
525 block->end = block->start + used_length;
528 /* Sort and search routines. */
530 /*--------------------------------------------------------------------------.
531 | Compare two words, FIRST and SECOND, and return 0 if they are identical. |
532 | Return less than 0 if the first word goes before the second; return |
533 | greater than 0 if the first word goes after the second. |
535 | If a word is indeed a prefix of the other, the shorter should go first. |
536 `--------------------------------------------------------------------------*/
538 static int
539 compare_words (const void *void_first, const void *void_second)
541 #define first ((const WORD *) void_first)
542 #define second ((const WORD *) void_second)
543 int length; /* minimum of two lengths */
544 int counter; /* cursor in words */
545 int value; /* value of comparison */
547 length = first->size < second->size ? first->size : second->size;
549 if (ignore_case)
551 for (counter = 0; counter < length; counter++)
553 value = (folded_chars [to_uchar (first->start[counter])]
554 - folded_chars [to_uchar (second->start[counter])]);
555 if (value != 0)
556 return value;
559 else
561 for (counter = 0; counter < length; counter++)
563 value = (to_uchar (first->start[counter])
564 - to_uchar (second->start[counter]));
565 if (value != 0)
566 return value;
570 return first->size - second->size;
571 #undef first
572 #undef second
575 /*-----------------------------------------------------------------------.
576 | Decides which of two OCCURS, FIRST or SECOND, should lexicographically |
577 | go first. In case of a tie, preserve the original order through a |
578 | pointer comparison. |
579 `-----------------------------------------------------------------------*/
581 static int
582 compare_occurs (const void *void_first, const void *void_second)
584 #define first ((const OCCURS *) void_first)
585 #define second ((const OCCURS *) void_second)
586 int value;
588 value = compare_words (&first->key, &second->key);
589 return value == 0 ? first->key.start - second->key.start : value;
590 #undef first
591 #undef second
594 /*------------------------------------------------------------.
595 | Return !0 if WORD appears in TABLE. Uses a binary search. |
596 `------------------------------------------------------------*/
598 static int _GL_ATTRIBUTE_PURE
599 search_table (WORD *word, WORD_TABLE *table)
601 int lowest; /* current lowest possible index */
602 int highest; /* current highest possible index */
603 int middle; /* current middle index */
604 int value; /* value from last comparison */
606 lowest = 0;
607 highest = table->length - 1;
608 while (lowest <= highest)
610 middle = (lowest + highest) / 2;
611 value = compare_words (word, table->start + middle);
612 if (value < 0)
613 highest = middle - 1;
614 else if (value > 0)
615 lowest = middle + 1;
616 else
617 return 1;
619 return 0;
622 /*---------------------------------------------------------------------.
623 | Sort the whole occurs table in memory. Presumably, 'qsort' does not |
624 | take intermediate copies or table elements, so the sort will be |
625 | stabilized throughout the comparison routine. |
626 `---------------------------------------------------------------------*/
628 static void
629 sort_found_occurs (void)
632 /* Only one language for the time being. */
633 if (number_of_occurs[0])
634 qsort (occurs_table[0], number_of_occurs[0], sizeof **occurs_table,
635 compare_occurs);
638 /* Parameter files reading routines. */
640 /*----------------------------------------------------------------------.
641 | Read a file named FILE_NAME, containing a set of break characters. |
642 | Build a content to the array word_fastmap in which all characters are |
643 | allowed except those found in the file. Characters may be repeated. |
644 `----------------------------------------------------------------------*/
646 static void
647 digest_break_file (const char *file_name)
649 BLOCK file_contents; /* to receive a copy of the file */
650 char *cursor; /* cursor in file copy */
652 swallow_file_in_memory (file_name, &file_contents);
654 /* Make the fastmap and record the file contents in it. */
656 memset (word_fastmap, 1, CHAR_SET_SIZE);
657 for (cursor = file_contents.start; cursor < file_contents.end; cursor++)
658 word_fastmap[to_uchar (*cursor)] = 0;
660 if (!gnu_extensions)
663 /* If GNU extensions are enabled, the only way to avoid newline as
664 a break character is to write all the break characters in the
665 file with no newline at all, not even at the end of the file.
666 If disabled, spaces, tabs and newlines are always considered as
667 break characters even if not included in the break file. */
669 word_fastmap[' '] = 0;
670 word_fastmap['\t'] = 0;
671 word_fastmap['\n'] = 0;
674 /* Return the space of the file, which is no more required. */
676 free (file_contents.start);
679 /*-----------------------------------------------------------------------.
680 | Read a file named FILE_NAME, containing one word per line, then |
681 | construct in TABLE a table of WORD descriptors for them. The routine |
682 | swallows the whole file in memory; this is at the expense of space |
683 | needed for newlines, which are useless; however, the reading is fast. |
684 `-----------------------------------------------------------------------*/
686 static void
687 digest_word_file (const char *file_name, WORD_TABLE *table)
689 BLOCK file_contents; /* to receive a copy of the file */
690 char *cursor; /* cursor in file copy */
691 char *word_start; /* start of the current word */
693 swallow_file_in_memory (file_name, &file_contents);
695 table->start = NULL;
696 table->alloc = 0;
697 table->length = 0;
699 /* Read the whole file. */
701 cursor = file_contents.start;
702 while (cursor < file_contents.end)
705 /* Read one line, and save the word in contains. */
707 word_start = cursor;
708 while (cursor < file_contents.end && *cursor != '\n')
709 cursor++;
711 /* Record the word in table if it is not empty. */
713 if (cursor > word_start)
715 if (table->length == table->alloc)
717 if ((SIZE_MAX / sizeof *table->start - 1) / 2 < table->alloc)
718 xalloc_die ();
719 table->alloc = table->alloc * 2 + 1;
720 table->start = xrealloc (table->start,
721 table->alloc * sizeof *table->start);
724 table->start[table->length].start = word_start;
725 table->start[table->length].size = cursor - word_start;
726 table->length++;
729 /* This test allows for an incomplete line at end of file. */
731 if (cursor < file_contents.end)
732 cursor++;
735 /* Finally, sort all the words read. */
737 qsort (table->start, table->length, sizeof table->start[0], compare_words);
740 /* Keyword recognition and selection. */
742 /*----------------------------------------------------------------------.
743 | For each keyword in the source text, constructs an OCCURS structure. |
744 `----------------------------------------------------------------------*/
746 static void
747 find_occurs_in_text (size_t file_index)
749 char *cursor; /* for scanning the source text */
750 char *scan; /* for scanning the source text also */
751 char *line_start; /* start of the current input line */
752 char *line_scan; /* newlines scanned until this point */
753 int reference_length; /* length of reference in input mode */
754 WORD possible_key; /* possible key, to ease searches */
755 OCCURS *occurs_cursor; /* current OCCURS under construction */
757 char *context_start; /* start of left context */
758 char *context_end; /* end of right context */
759 char *word_start; /* start of word */
760 char *word_end; /* end of word */
761 char *next_context_start; /* next start of left context */
763 const BLOCK *text_buffer = &text_buffers[file_index];
765 /* reference_length is always used within 'if (input_reference)'.
766 However, GNU C diagnoses that it may be used uninitialized. The
767 following assignment is merely to shut it up. */
769 reference_length = 0;
771 /* Tracking where lines start is helpful for reference processing. In
772 auto reference mode, this allows counting lines. In input reference
773 mode, this permits finding the beginning of the references.
775 The first line begins with the file, skip immediately this very first
776 reference in input reference mode, to help further rejection any word
777 found inside it. Also, unconditionally assigning these variable has
778 the happy effect of shutting up lint. */
780 line_start = text_buffer->start;
781 line_scan = line_start;
782 if (input_reference)
784 SKIP_NON_WHITE (line_scan, text_buffer->end);
785 reference_length = line_scan - line_start;
786 SKIP_WHITE (line_scan, text_buffer->end);
789 /* Process the whole buffer, one line or one sentence at a time. */
791 for (cursor = text_buffer->start;
792 cursor < text_buffer->end;
793 cursor = next_context_start)
796 /* 'context_start' gets initialized before the processing of each
797 line, or once for the whole buffer if no end of line or sentence
798 sequence separator. */
800 context_start = cursor;
802 /* If an end of line or end of sentence sequence is defined and
803 non-empty, 'next_context_start' will be recomputed to be the end of
804 each line or sentence, before each one is processed. If no such
805 sequence, then 'next_context_start' is set at the end of the whole
806 buffer, which is then considered to be a single line or sentence.
807 This test also accounts for the case of an incomplete line or
808 sentence at the end of the buffer. */
810 next_context_start = text_buffer->end;
811 if (context_regex.string)
812 switch (re_search (&context_regex.pattern, cursor,
813 text_buffer->end - cursor,
814 0, text_buffer->end - cursor, &context_regs))
816 case -2:
817 matcher_error ();
819 case -1:
820 break;
822 default:
823 next_context_start = cursor + context_regs.end[0];
824 break;
827 /* Include the separator into the right context, but not any suffix
828 white space in this separator; this insures it will be seen in
829 output and will not take more space than necessary. */
831 context_end = next_context_start;
832 SKIP_WHITE_BACKWARDS (context_end, context_start);
834 /* Read and process a single input line or sentence, one word at a
835 time. */
837 while (1)
839 if (word_regex.string)
841 /* If a word regexp has been compiled, use it to skip at the
842 beginning of the next word. If there is no such word, exit
843 the loop. */
846 regoff_t r = re_search (&word_regex.pattern, cursor,
847 context_end - cursor,
848 0, context_end - cursor, &word_regs);
849 if (r == -2)
850 matcher_error ();
851 if (r == -1)
852 break;
853 word_start = cursor + word_regs.start[0];
854 word_end = cursor + word_regs.end[0];
856 else
858 /* Avoid re_search and use the fastmap to skip to the
859 beginning of the next word. If there is no more word in
860 the buffer, exit the loop. */
863 scan = cursor;
864 while (scan < context_end
865 && !word_fastmap[to_uchar (*scan)])
866 scan++;
868 if (scan == context_end)
869 break;
871 word_start = scan;
873 while (scan < context_end
874 && word_fastmap[to_uchar (*scan)])
875 scan++;
877 word_end = scan;
880 /* Skip right to the beginning of the found word. */
882 cursor = word_start;
884 /* Skip any zero length word. Just advance a single position,
885 then go fetch the next word. */
887 if (word_end == word_start)
889 cursor++;
890 continue;
893 /* This is a genuine, non empty word, so save it as a possible
894 key. Then skip over it. Also, maintain the maximum length of
895 all words read so far. It is mandatory to take the maximum
896 length of all words in the file, without considering if they
897 are actually kept or rejected, because backward jumps at output
898 generation time may fall in *any* word. */
900 possible_key.start = cursor;
901 possible_key.size = word_end - word_start;
902 cursor += possible_key.size;
904 if (possible_key.size > maximum_word_length)
905 maximum_word_length = possible_key.size;
907 /* In input reference mode, update 'line_start' from its previous
908 value. Count the lines just in case auto reference mode is
909 also selected. If it happens that the word just matched is
910 indeed part of a reference; just ignore it. */
912 if (input_reference)
914 while (line_scan < possible_key.start)
915 if (*line_scan == '\n')
917 total_line_count++;
918 line_scan++;
919 line_start = line_scan;
920 SKIP_NON_WHITE (line_scan, text_buffer->end);
921 reference_length = line_scan - line_start;
923 else
924 line_scan++;
925 if (line_scan > possible_key.start)
926 continue;
929 /* Ignore the word if an 'Ignore words' table exists and if it is
930 part of it. Also ignore the word if an 'Only words' table and
931 if it is *not* part of it.
933 It is allowed that both tables be used at once, even if this
934 may look strange for now. Just ignore a word that would appear
935 in both. If regexps are eventually implemented for these
936 tables, the Ignore table could then reject words that would
937 have been previously accepted by the Only table. */
939 if (ignore_file && search_table (&possible_key, &ignore_table))
940 continue;
941 if (only_file && !search_table (&possible_key, &only_table))
942 continue;
944 /* A non-empty word has been found. First of all, insure
945 proper allocation of the next OCCURS, and make a pointer to
946 where it will be constructed. */
948 if (number_of_occurs[0] == occurs_alloc[0])
950 if ((SIZE_MAX / sizeof *occurs_table[0] - 1) / 2
951 < occurs_alloc[0])
952 xalloc_die ();
953 occurs_alloc[0] = occurs_alloc[0] * 2 + 1;
954 occurs_table[0] =
955 xrealloc (occurs_table[0],
956 occurs_alloc[0] * sizeof *occurs_table[0]);
959 occurs_cursor = occurs_table[0] + number_of_occurs[0];
961 /* Define the reference field, if any. */
963 if (auto_reference)
966 /* While auto referencing, update 'line_start' from its
967 previous value, counting lines as we go. If input
968 referencing at the same time, 'line_start' has been
969 advanced earlier, and the following loop is never really
970 executed. */
972 while (line_scan < possible_key.start)
973 if (*line_scan == '\n')
975 total_line_count++;
976 line_scan++;
977 line_start = line_scan;
978 SKIP_NON_WHITE (line_scan, text_buffer->end);
980 else
981 line_scan++;
983 occurs_cursor->reference = total_line_count;
985 else if (input_reference)
988 /* If only input referencing, 'line_start' has been computed
989 earlier to detect the case the word matched would be part
990 of the reference. The reference position is simply the
991 value of 'line_start'. */
993 occurs_cursor->reference
994 = (DELTA) (line_start - possible_key.start);
995 if (reference_length > reference_max_width)
996 reference_max_width = reference_length;
999 /* Exclude the reference from the context in simple cases. */
1001 if (input_reference && line_start == context_start)
1003 SKIP_NON_WHITE (context_start, context_end);
1004 SKIP_WHITE (context_start, context_end);
1007 /* Completes the OCCURS structure. */
1009 occurs_cursor->key = possible_key;
1010 occurs_cursor->left = context_start - possible_key.start;
1011 occurs_cursor->right = context_end - possible_key.start;
1012 occurs_cursor->file_index = file_index;
1014 number_of_occurs[0]++;
1019 /* Formatting and actual output - service routines. */
1021 /*-----------------------------------------.
1022 | Prints some NUMBER of spaces on stdout. |
1023 `-----------------------------------------*/
1025 static void
1026 print_spaces (int number)
1028 int counter;
1030 for (counter = number; counter > 0; counter--)
1031 putchar (' ');
1034 /*-------------------------------------.
1035 | Prints the field provided by FIELD. |
1036 `-------------------------------------*/
1038 static void
1039 print_field (BLOCK field)
1041 char *cursor; /* Cursor in field to print */
1042 int base; /* Base character, without diacritic */
1043 int diacritic; /* Diacritic code for the character */
1045 /* Whitespace is not really compressed. Instead, each white space
1046 character (tab, vt, ht etc.) is printed as one single space. */
1048 for (cursor = field.start; cursor < field.end; cursor++)
1050 unsigned char character = *cursor;
1051 if (edited_flag[character])
1054 /* First check if this is a diacriticized character.
1056 This works only for TeX. I do not know how diacriticized
1057 letters work with 'roff'. Please someone explain it to me! */
1059 diacritic = todiac (character);
1060 if (diacritic != 0 && output_format == TEX_FORMAT)
1062 base = tobase (character);
1063 switch (diacritic)
1066 case 1: /* Latin diphthongs */
1067 switch (base)
1069 case 'o':
1070 fputs ("\\oe{}", stdout);
1071 break;
1073 case 'O':
1074 fputs ("\\OE{}", stdout);
1075 break;
1077 case 'a':
1078 fputs ("\\ae{}", stdout);
1079 break;
1081 case 'A':
1082 fputs ("\\AE{}", stdout);
1083 break;
1085 default:
1086 putchar (' ');
1088 break;
1090 case 2: /* Acute accent */
1091 printf ("\\'%s%c", (base == 'i' ? "\\" : ""), base);
1092 break;
1094 case 3: /* Grave accent */
1095 printf ("\\'%s%c", (base == 'i' ? "\\" : ""), base);
1096 break;
1098 case 4: /* Circumflex accent */
1099 printf ("\\^%s%c", (base == 'i' ? "\\" : ""), base);
1100 break;
1102 case 5: /* Diaeresis */
1103 printf ("\\\"%s%c", (base == 'i' ? "\\" : ""), base);
1104 break;
1106 case 6: /* Tilde accent */
1107 printf ("\\~%s%c", (base == 'i' ? "\\" : ""), base);
1108 break;
1110 case 7: /* Cedilla */
1111 printf ("\\c{%c}", base);
1112 break;
1114 case 8: /* Small circle beneath */
1115 switch (base)
1117 case 'a':
1118 fputs ("\\aa{}", stdout);
1119 break;
1121 case 'A':
1122 fputs ("\\AA{}", stdout);
1123 break;
1125 default:
1126 putchar (' ');
1128 break;
1130 case 9: /* Strike through */
1131 switch (base)
1133 case 'o':
1134 fputs ("\\o{}", stdout);
1135 break;
1137 case 'O':
1138 fputs ("\\O{}", stdout);
1139 break;
1141 default:
1142 putchar (' ');
1144 break;
1147 else
1149 /* This is not a diacritic character, so handle cases which are
1150 really specific to 'roff' or TeX. All white space processing
1151 is done as the default case of this switch. */
1153 switch (character)
1155 case '"':
1156 /* In roff output format, double any quote. */
1157 putchar ('"');
1158 putchar ('"');
1159 break;
1161 case '$':
1162 case '%':
1163 case '&':
1164 case '#':
1165 case '_':
1166 /* In TeX output format, precede these with a backslash. */
1167 putchar ('\\');
1168 putchar (character);
1169 break;
1171 case '{':
1172 case '}':
1173 /* In TeX output format, precede these with a backslash and
1174 force mathematical mode. */
1175 printf ("$\\%c$", character);
1176 break;
1178 case '\\':
1179 /* In TeX output mode, request production of a backslash. */
1180 fputs ("\\backslash{}", stdout);
1181 break;
1183 default:
1184 /* Any other flagged character produces a single space. */
1185 putchar (' ');
1188 else
1189 putchar (*cursor);
1193 /* Formatting and actual output - planning routines. */
1195 /*--------------------------------------------------------------------.
1196 | From information collected from command line options and input file |
1197 | readings, compute and fix some output parameter values. |
1198 `--------------------------------------------------------------------*/
1200 static void
1201 fix_output_parameters (void)
1203 size_t file_index; /* index in text input file arrays */
1204 int line_ordinal; /* line ordinal value for reference */
1205 char ordinal_string[12]; /* edited line ordinal for reference */
1206 int reference_width; /* width for the whole reference */
1207 int character; /* character ordinal */
1208 const char *cursor; /* cursor in some constant strings */
1210 /* In auto reference mode, the maximum width of this field is
1211 precomputed and subtracted from the overall line width. Add one for
1212 the column which separate the file name from the line number. */
1214 if (auto_reference)
1216 reference_max_width = 0;
1217 for (file_index = 0; file_index < number_input_files; file_index++)
1219 line_ordinal = file_line_count[file_index] + 1;
1220 if (file_index > 0)
1221 line_ordinal -= file_line_count[file_index - 1];
1222 sprintf (ordinal_string, "%d", line_ordinal);
1223 reference_width = strlen (ordinal_string);
1224 if (input_file_name[file_index])
1225 reference_width += strlen (input_file_name[file_index]);
1226 if (reference_width > reference_max_width)
1227 reference_max_width = reference_width;
1229 reference_max_width++;
1230 reference.start = xmalloc ((size_t) reference_max_width + 1);
1233 /* If the reference appears to the left of the output line, reserve some
1234 space for it right away, including one gap size. */
1236 if ((auto_reference || input_reference) && !right_reference)
1237 line_width -= reference_max_width + gap_size;
1238 if (line_width < 0)
1239 line_width = 0;
1241 /* The output lines, minimally, will contain from left to right a left
1242 context, a gap, and a keyword followed by the right context with no
1243 special intervening gap. Half of the line width is dedicated to the
1244 left context and the gap, the other half is dedicated to the keyword
1245 and the right context; these values are computed once and for all here.
1246 There also are tail and head wrap around fields, used when the keyword
1247 is near the beginning or the end of the line, or when some long word
1248 cannot fit in, but leave place from wrapped around shorter words. The
1249 maximum width of these fields are recomputed separately for each line,
1250 on a case by case basis. It is worth noting that it cannot happen that
1251 both the tail and head fields are used at once. */
1253 half_line_width = line_width / 2;
1254 before_max_width = half_line_width - gap_size;
1255 keyafter_max_width = half_line_width;
1257 /* If truncation_string is the empty string, make it NULL to speed up
1258 tests. In this case, truncation_string_length will never get used, so
1259 there is no need to set it. */
1261 if (truncation_string && *truncation_string)
1262 truncation_string_length = strlen (truncation_string);
1263 else
1264 truncation_string = NULL;
1266 if (gnu_extensions)
1269 /* When flagging truncation at the left of the keyword, the
1270 truncation mark goes at the beginning of the before field,
1271 unless there is a head field, in which case the mark goes at the
1272 left of the head field. When flagging truncation at the right
1273 of the keyword, the mark goes at the end of the keyafter field,
1274 unless there is a tail field, in which case the mark goes at the
1275 end of the tail field. Only eight combination cases could arise
1276 for truncation marks:
1278 . None.
1279 . One beginning the before field.
1280 . One beginning the head field.
1281 . One ending the keyafter field.
1282 . One ending the tail field.
1283 . One beginning the before field, another ending the keyafter field.
1284 . One ending the tail field, another beginning the before field.
1285 . One ending the keyafter field, another beginning the head field.
1287 So, there is at most two truncation marks, which could appear both
1288 on the left side of the center of the output line, both on the
1289 right side, or one on either side. */
1291 before_max_width -= 2 * truncation_string_length;
1292 if (before_max_width < 0)
1293 before_max_width = 0;
1294 keyafter_max_width -= 2 * truncation_string_length;
1296 else
1299 /* I never figured out exactly how UNIX' ptx plans the output width
1300 of its various fields. If GNU extensions are disabled, do not
1301 try computing the field widths correctly; instead, use the
1302 following formula, which does not completely imitate UNIX' ptx,
1303 but almost. */
1305 keyafter_max_width -= 2 * truncation_string_length + 1;
1308 /* Compute which characters need special output processing. Initialize
1309 by flagging any white space character. Some systems do not consider
1310 form feed as a space character, but we do. */
1312 for (character = 0; character < CHAR_SET_SIZE; character++)
1313 edited_flag[character] = !! isspace (character);
1314 edited_flag['\f'] = 1;
1316 /* Complete the special character flagging according to selected output
1317 format. */
1319 switch (output_format)
1321 case UNKNOWN_FORMAT:
1322 /* Should never happen. */
1324 case DUMB_FORMAT:
1325 break;
1327 case ROFF_FORMAT:
1329 /* 'Quote' characters should be doubled. */
1331 edited_flag['"'] = 1;
1332 break;
1334 case TEX_FORMAT:
1336 /* Various characters need special processing. */
1338 for (cursor = "$%&#_{}\\"; *cursor; cursor++)
1339 edited_flag[to_uchar (*cursor)] = 1;
1341 /* Any character with 8th bit set will print to a single space, unless
1342 it is diacriticized. */
1344 for (character = 0200; character < CHAR_SET_SIZE; character++)
1345 edited_flag[character] = todiac (character) != 0;
1346 break;
1350 /*------------------------------------------------------------------.
1351 | Compute the position and length of all the output fields, given a |
1352 | pointer to some OCCURS. |
1353 `------------------------------------------------------------------*/
1355 static void
1356 define_all_fields (OCCURS *occurs)
1358 int tail_max_width; /* allowable width of tail field */
1359 int head_max_width; /* allowable width of head field */
1360 char *cursor; /* running cursor in source text */
1361 char *left_context_start; /* start of left context */
1362 char *right_context_end; /* end of right context */
1363 char *left_field_start; /* conservative start for 'head'/'before' */
1364 const char *file_name; /* file name for reference */
1365 int line_ordinal; /* line ordinal for reference */
1366 const char *buffer_start; /* start of buffered file for this occurs */
1367 const char *buffer_end; /* end of buffered file for this occurs */
1369 /* Define 'keyafter', start of left context and end of right context.
1370 'keyafter' starts at the saved position for keyword and extend to the
1371 right from the end of the keyword, eating separators or full words, but
1372 not beyond maximum allowed width for 'keyafter' field or limit for the
1373 right context. Suffix spaces will be removed afterwards. */
1375 keyafter.start = occurs->key.start;
1376 keyafter.end = keyafter.start + occurs->key.size;
1377 left_context_start = keyafter.start + occurs->left;
1378 right_context_end = keyafter.start + occurs->right;
1380 buffer_start = text_buffers[occurs->file_index].start;
1381 buffer_end = text_buffers[occurs->file_index].end;
1383 cursor = keyafter.end;
1384 while (cursor < right_context_end
1385 && cursor <= keyafter.start + keyafter_max_width)
1387 keyafter.end = cursor;
1388 SKIP_SOMETHING (cursor, right_context_end);
1390 if (cursor <= keyafter.start + keyafter_max_width)
1391 keyafter.end = cursor;
1393 keyafter_truncation = truncation_string && keyafter.end < right_context_end;
1395 SKIP_WHITE_BACKWARDS (keyafter.end, keyafter.start);
1397 /* When the left context is wide, it might take some time to catch up from
1398 the left context boundary to the beginning of the 'head' or 'before'
1399 fields. So, in this case, to speed the catchup, we jump back from the
1400 keyword, using some secure distance, possibly falling in the middle of
1401 a word. A secure backward jump would be at least half the maximum
1402 width of a line, plus the size of the longest word met in the whole
1403 input. We conclude this backward jump by a skip forward of at least
1404 one word. In this manner, we should not inadvertently accept only part
1405 of a word. From the reached point, when it will be time to fix the
1406 beginning of 'head' or 'before' fields, we will skip forward words or
1407 delimiters until we get sufficiently near. */
1409 if (-occurs->left > half_line_width + maximum_word_length)
1411 left_field_start
1412 = keyafter.start - (half_line_width + maximum_word_length);
1413 SKIP_SOMETHING (left_field_start, keyafter.start);
1415 else
1416 left_field_start = keyafter.start + occurs->left;
1418 /* 'before' certainly ends at the keyword, but not including separating
1419 spaces. It starts after than the saved value for the left context, by
1420 advancing it until it falls inside the maximum allowed width for the
1421 before field. There will be no prefix spaces either. 'before' only
1422 advances by skipping single separators or whole words. */
1424 before.start = left_field_start;
1425 before.end = keyafter.start;
1426 SKIP_WHITE_BACKWARDS (before.end, before.start);
1428 while (before.start + before_max_width < before.end)
1429 SKIP_SOMETHING (before.start, before.end);
1431 if (truncation_string)
1433 cursor = before.start;
1434 SKIP_WHITE_BACKWARDS (cursor, buffer_start);
1435 before_truncation = cursor > left_context_start;
1437 else
1438 before_truncation = 0;
1440 SKIP_WHITE (before.start, buffer_end);
1442 /* The tail could not take more columns than what has been left in the
1443 left context field, and a gap is mandatory. It starts after the
1444 right context, and does not contain prefixed spaces. It ends at
1445 the end of line, the end of buffer or when the tail field is full,
1446 whichever comes first. It cannot contain only part of a word, and
1447 has no suffixed spaces. */
1449 tail_max_width
1450 = before_max_width - (before.end - before.start) - gap_size;
1452 if (tail_max_width > 0)
1454 tail.start = keyafter.end;
1455 SKIP_WHITE (tail.start, buffer_end);
1457 tail.end = tail.start;
1458 cursor = tail.end;
1459 while (cursor < right_context_end
1460 && cursor < tail.start + tail_max_width)
1462 tail.end = cursor;
1463 SKIP_SOMETHING (cursor, right_context_end);
1466 if (cursor < tail.start + tail_max_width)
1467 tail.end = cursor;
1469 if (tail.end > tail.start)
1471 keyafter_truncation = 0;
1472 tail_truncation = truncation_string && tail.end < right_context_end;
1474 else
1475 tail_truncation = 0;
1477 SKIP_WHITE_BACKWARDS (tail.end, tail.start);
1479 else
1482 /* No place left for a tail field. */
1484 tail.start = NULL;
1485 tail.end = NULL;
1486 tail_truncation = 0;
1489 /* 'head' could not take more columns than what has been left in the right
1490 context field, and a gap is mandatory. It ends before the left
1491 context, and does not contain suffixed spaces. Its pointer is advanced
1492 until the head field has shrunk to its allowed width. It cannot
1493 contain only part of a word, and has no suffixed spaces. */
1495 head_max_width
1496 = keyafter_max_width - (keyafter.end - keyafter.start) - gap_size;
1498 if (head_max_width > 0)
1500 head.end = before.start;
1501 SKIP_WHITE_BACKWARDS (head.end, buffer_start);
1503 head.start = left_field_start;
1504 while (head.start + head_max_width < head.end)
1505 SKIP_SOMETHING (head.start, head.end);
1507 if (head.end > head.start)
1509 before_truncation = 0;
1510 head_truncation = (truncation_string
1511 && head.start > left_context_start);
1513 else
1514 head_truncation = 0;
1516 SKIP_WHITE (head.start, head.end);
1518 else
1521 /* No place left for a head field. */
1523 head.start = NULL;
1524 head.end = NULL;
1525 head_truncation = 0;
1528 if (auto_reference)
1531 /* Construct the reference text in preallocated space from the file
1532 name and the line number. Standard input yields an empty file name.
1533 Ensure line numbers are 1 based, even if they are computed 0 based. */
1535 file_name = input_file_name[occurs->file_index];
1536 if (!file_name)
1537 file_name = "";
1539 line_ordinal = occurs->reference + 1;
1540 if (occurs->file_index > 0)
1541 line_ordinal -= file_line_count[occurs->file_index - 1];
1543 sprintf (reference.start, "%s:%d", file_name, line_ordinal);
1544 reference.end = reference.start + strlen (reference.start);
1546 else if (input_reference)
1549 /* Reference starts at saved position for reference and extends right
1550 until some white space is met. */
1552 reference.start = keyafter.start + (DELTA) occurs->reference;
1553 reference.end = reference.start;
1554 SKIP_NON_WHITE (reference.end, right_context_end);
1558 /* Formatting and actual output - control routines. */
1560 /*----------------------------------------------------------------------.
1561 | Output the current output fields as one line for 'troff' or 'nroff'. |
1562 `----------------------------------------------------------------------*/
1564 static void
1565 output_one_roff_line (void)
1567 /* Output the 'tail' field. */
1569 printf (".%s \"", macro_name);
1570 print_field (tail);
1571 if (tail_truncation)
1572 fputs (truncation_string, stdout);
1573 putchar ('"');
1575 /* Output the 'before' field. */
1577 fputs (" \"", stdout);
1578 if (before_truncation)
1579 fputs (truncation_string, stdout);
1580 print_field (before);
1581 putchar ('"');
1583 /* Output the 'keyafter' field. */
1585 fputs (" \"", stdout);
1586 print_field (keyafter);
1587 if (keyafter_truncation)
1588 fputs (truncation_string, stdout);
1589 putchar ('"');
1591 /* Output the 'head' field. */
1593 fputs (" \"", stdout);
1594 if (head_truncation)
1595 fputs (truncation_string, stdout);
1596 print_field (head);
1597 putchar ('"');
1599 /* Conditionally output the 'reference' field. */
1601 if (auto_reference || input_reference)
1603 fputs (" \"", stdout);
1604 print_field (reference);
1605 putchar ('"');
1608 putchar ('\n');
1611 /*---------------------------------------------------------.
1612 | Output the current output fields as one line for 'TeX'. |
1613 `---------------------------------------------------------*/
1615 static void
1616 output_one_tex_line (void)
1618 BLOCK key; /* key field, isolated */
1619 BLOCK after; /* after field, isolated */
1620 char *cursor; /* running cursor in source text */
1622 printf ("\\%s ", macro_name);
1623 putchar ('{');
1624 print_field (tail);
1625 fputs ("}{", stdout);
1626 print_field (before);
1627 fputs ("}{", stdout);
1628 key.start = keyafter.start;
1629 after.end = keyafter.end;
1630 cursor = keyafter.start;
1631 SKIP_SOMETHING (cursor, keyafter.end);
1632 key.end = cursor;
1633 after.start = cursor;
1634 print_field (key);
1635 fputs ("}{", stdout);
1636 print_field (after);
1637 fputs ("}{", stdout);
1638 print_field (head);
1639 putchar ('}');
1640 if (auto_reference || input_reference)
1642 putchar ('{');
1643 print_field (reference);
1644 putchar ('}');
1646 putchar ('\n');
1649 /*-------------------------------------------------------------------.
1650 | Output the current output fields as one line for a dumb terminal. |
1651 `-------------------------------------------------------------------*/
1653 static void
1654 output_one_dumb_line (void)
1656 if (!right_reference)
1658 if (auto_reference)
1661 /* Output the 'reference' field, in such a way that GNU emacs
1662 next-error will handle it. The ending colon is taken from the
1663 gap which follows. */
1665 print_field (reference);
1666 putchar (':');
1667 print_spaces (reference_max_width
1668 + gap_size
1669 - (reference.end - reference.start)
1670 - 1);
1672 else
1675 /* Output the 'reference' field and its following gap. */
1677 print_field (reference);
1678 print_spaces (reference_max_width
1679 + gap_size
1680 - (reference.end - reference.start));
1684 if (tail.start < tail.end)
1686 /* Output the 'tail' field. */
1688 print_field (tail);
1689 if (tail_truncation)
1690 fputs (truncation_string, stdout);
1692 print_spaces (half_line_width - gap_size
1693 - (before.end - before.start)
1694 - (before_truncation ? truncation_string_length : 0)
1695 - (tail.end - tail.start)
1696 - (tail_truncation ? truncation_string_length : 0));
1698 else
1699 print_spaces (half_line_width - gap_size
1700 - (before.end - before.start)
1701 - (before_truncation ? truncation_string_length : 0));
1703 /* Output the 'before' field. */
1705 if (before_truncation)
1706 fputs (truncation_string, stdout);
1707 print_field (before);
1709 print_spaces (gap_size);
1711 /* Output the 'keyafter' field. */
1713 print_field (keyafter);
1714 if (keyafter_truncation)
1715 fputs (truncation_string, stdout);
1717 if (head.start < head.end)
1719 /* Output the 'head' field. */
1721 print_spaces (half_line_width
1722 - (keyafter.end - keyafter.start)
1723 - (keyafter_truncation ? truncation_string_length : 0)
1724 - (head.end - head.start)
1725 - (head_truncation ? truncation_string_length : 0));
1726 if (head_truncation)
1727 fputs (truncation_string, stdout);
1728 print_field (head);
1730 else
1732 if ((auto_reference || input_reference) && right_reference)
1733 print_spaces (half_line_width
1734 - (keyafter.end - keyafter.start)
1735 - (keyafter_truncation ? truncation_string_length : 0));
1737 if ((auto_reference || input_reference) && right_reference)
1739 /* Output the 'reference' field. */
1741 print_spaces (gap_size);
1742 print_field (reference);
1745 putchar ('\n');
1748 /*------------------------------------------------------------------------.
1749 | Scan the whole occurs table and, for each entry, output one line in the |
1750 | appropriate format. |
1751 `------------------------------------------------------------------------*/
1753 static void
1754 generate_all_output (void)
1756 size_t occurs_index; /* index of keyword entry being processed */
1757 OCCURS *occurs_cursor; /* current keyword entry being processed */
1759 /* The following assignments are useful to provide default values in case
1760 line contexts or references are not used, in which case these variables
1761 would never be computed. */
1763 tail.start = NULL;
1764 tail.end = NULL;
1765 tail_truncation = 0;
1767 head.start = NULL;
1768 head.end = NULL;
1769 head_truncation = 0;
1771 /* Loop over all keyword occurrences. */
1773 occurs_cursor = occurs_table[0];
1775 for (occurs_index = 0; occurs_index < number_of_occurs[0]; occurs_index++)
1777 /* Compute the exact size of every field and whenever truncation flags
1778 are present or not. */
1780 define_all_fields (occurs_cursor);
1782 /* Produce one output line according to selected format. */
1784 switch (output_format)
1786 case UNKNOWN_FORMAT:
1787 /* Should never happen. */
1789 case DUMB_FORMAT:
1790 output_one_dumb_line ();
1791 break;
1793 case ROFF_FORMAT:
1794 output_one_roff_line ();
1795 break;
1797 case TEX_FORMAT:
1798 output_one_tex_line ();
1799 break;
1802 /* Advance the cursor into the occurs table. */
1804 occurs_cursor++;
1808 /* Option decoding and main program. */
1810 /*------------------------------------------------------.
1811 | Print program identification and options, then exit. |
1812 `------------------------------------------------------*/
1814 void
1815 usage (int status)
1817 if (status != EXIT_SUCCESS)
1818 emit_try_help ();
1819 else
1821 printf (_("\
1822 Usage: %s [OPTION]... [INPUT]... (without -G)\n\
1823 or: %s -G [OPTION]... [INPUT [OUTPUT]]\n"),
1824 program_name, program_name);
1825 fputs (_("\
1826 Output a permuted index, including context, of the words in the input files.\n\
1827 "), stdout);
1829 emit_stdin_note ();
1830 emit_mandatory_arg_note ();
1832 fputs (_("\
1833 -A, --auto-reference output automatically generated references\n\
1834 -G, --traditional behave more like System V 'ptx'\n\
1835 "), stdout);
1836 fputs (_("\
1837 -F, --flag-truncation=STRING use STRING for flagging line truncations.\n\
1838 The default is '/'\n\
1839 "), stdout);
1840 fputs (_("\
1841 -M, --macro-name=STRING macro name to use instead of 'xx'\n\
1842 -O, --format=roff generate output as roff directives\n\
1843 -R, --right-side-refs put references at right, not counted in -w\n\
1844 -S, --sentence-regexp=REGEXP for end of lines or end of sentences\n\
1845 -T, --format=tex generate output as TeX directives\n\
1846 "), stdout);
1847 fputs (_("\
1848 -W, --word-regexp=REGEXP use REGEXP to match each keyword\n\
1849 -b, --break-file=FILE word break characters in this FILE\n\
1850 -f, --ignore-case fold lower case to upper case for sorting\n\
1851 -g, --gap-size=NUMBER gap size in columns between output fields\n\
1852 -i, --ignore-file=FILE read ignore word list from FILE\n\
1853 -o, --only-file=FILE read only word list from this FILE\n\
1854 "), stdout);
1855 fputs (_("\
1856 -r, --references first field of each line is a reference\n\
1857 -t, --typeset-mode - not implemented -\n\
1858 -w, --width=NUMBER output width in columns, reference excluded\n\
1859 "), stdout);
1860 fputs (HELP_OPTION_DESCRIPTION, stdout);
1861 fputs (VERSION_OPTION_DESCRIPTION, stdout);
1862 emit_ancillary_info (PROGRAM_NAME);
1864 exit (status);
1867 /*----------------------------------------------------------------------.
1868 | Main program. Decode ARGC arguments passed through the ARGV array of |
1869 | strings, then launch execution. |
1870 `----------------------------------------------------------------------*/
1872 /* Long options equivalences. */
1873 static struct option const long_options[] =
1875 {"auto-reference", no_argument, NULL, 'A'},
1876 {"break-file", required_argument, NULL, 'b'},
1877 {"flag-truncation", required_argument, NULL, 'F'},
1878 {"ignore-case", no_argument, NULL, 'f'},
1879 {"gap-size", required_argument, NULL, 'g'},
1880 {"ignore-file", required_argument, NULL, 'i'},
1881 {"macro-name", required_argument, NULL, 'M'},
1882 {"only-file", required_argument, NULL, 'o'},
1883 {"references", no_argument, NULL, 'r'},
1884 {"right-side-refs", no_argument, NULL, 'R'},
1885 {"format", required_argument, NULL, 10},
1886 {"sentence-regexp", required_argument, NULL, 'S'},
1887 {"traditional", no_argument, NULL, 'G'},
1888 {"typeset-mode", no_argument, NULL, 't'},
1889 {"width", required_argument, NULL, 'w'},
1890 {"word-regexp", required_argument, NULL, 'W'},
1891 {GETOPT_HELP_OPTION_DECL},
1892 {GETOPT_VERSION_OPTION_DECL},
1893 {NULL, 0, NULL, 0},
1896 static char const* const format_args[] =
1898 "roff", "tex", NULL
1901 static enum Format const format_vals[] =
1903 ROFF_FORMAT, TEX_FORMAT
1907 main (int argc, char **argv)
1909 int optchar; /* argument character */
1910 int file_index; /* index in text input file arrays */
1912 /* Decode program options. */
1914 initialize_main (&argc, &argv);
1915 set_program_name (argv[0]);
1916 setlocale (LC_ALL, "");
1917 bindtextdomain (PACKAGE, LOCALEDIR);
1918 textdomain (PACKAGE);
1920 atexit (close_stdout);
1922 #if HAVE_SETCHRCLASS
1923 setchrclass (NULL);
1924 #endif
1926 while (optchar = getopt_long (argc, argv, "AF:GM:ORS:TW:b:i:fg:o:trw:",
1927 long_options, NULL),
1928 optchar != EOF)
1930 switch (optchar)
1932 default:
1933 usage (EXIT_FAILURE);
1935 case 'G':
1936 gnu_extensions = false;
1937 break;
1939 case 'b':
1940 break_file = optarg;
1941 break;
1943 case 'f':
1944 ignore_case = true;
1945 break;
1947 case 'g':
1949 unsigned long int tmp_ulong;
1950 if (xstrtoul (optarg, NULL, 0, &tmp_ulong, NULL) != LONGINT_OK
1951 || ! (0 < tmp_ulong && tmp_ulong <= INT_MAX))
1952 die (EXIT_FAILURE, 0, _("invalid gap width: %s"),
1953 quote (optarg));
1954 gap_size = tmp_ulong;
1955 break;
1958 case 'i':
1959 ignore_file = optarg;
1960 break;
1962 case 'o':
1963 only_file = optarg;
1964 break;
1966 case 'r':
1967 input_reference = true;
1968 break;
1970 case 't':
1971 /* Yet to understand... */
1972 break;
1974 case 'w':
1976 unsigned long int tmp_ulong;
1977 if (xstrtoul (optarg, NULL, 0, &tmp_ulong, NULL) != LONGINT_OK
1978 || ! (0 < tmp_ulong && tmp_ulong <= INT_MAX))
1979 die (EXIT_FAILURE, 0, _("invalid line width: %s"),
1980 quote (optarg));
1981 line_width = tmp_ulong;
1982 break;
1985 case 'A':
1986 auto_reference = true;
1987 break;
1989 case 'F':
1990 truncation_string = copy_unescaped_string (optarg);
1991 break;
1993 case 'M':
1994 macro_name = optarg;
1995 break;
1997 case 'O':
1998 output_format = ROFF_FORMAT;
1999 break;
2001 case 'R':
2002 right_reference = true;
2003 break;
2005 case 'S':
2006 context_regex.string = copy_unescaped_string (optarg);
2007 break;
2009 case 'T':
2010 output_format = TEX_FORMAT;
2011 break;
2013 case 'W':
2014 word_regex.string = copy_unescaped_string (optarg);
2015 if (!*word_regex.string)
2016 word_regex.string = NULL;
2017 break;
2019 case 10:
2020 output_format = XARGMATCH ("--format", optarg,
2021 format_args, format_vals);
2022 break;
2024 case_GETOPT_HELP_CHAR;
2026 case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS);
2030 /* Process remaining arguments. If GNU extensions are enabled, process
2031 all arguments as input parameters. If disabled, accept at most two
2032 arguments, the second of which is an output parameter. */
2034 if (optind == argc)
2037 /* No more argument simply means: read standard input. */
2039 input_file_name = xmalloc (sizeof *input_file_name);
2040 file_line_count = xmalloc (sizeof *file_line_count);
2041 text_buffers = xmalloc (sizeof *text_buffers);
2042 number_input_files = 1;
2043 input_file_name[0] = NULL;
2045 else if (gnu_extensions)
2047 number_input_files = argc - optind;
2048 input_file_name = xmalloc (number_input_files * sizeof *input_file_name);
2049 file_line_count = xmalloc (number_input_files * sizeof *file_line_count);
2050 text_buffers = xmalloc (number_input_files * sizeof *text_buffers);
2052 for (file_index = 0; file_index < number_input_files; file_index++)
2054 if (!*argv[optind] || STREQ (argv[optind], "-"))
2055 input_file_name[file_index] = NULL;
2056 else
2057 input_file_name[file_index] = argv[optind];
2058 optind++;
2061 else
2064 /* There is one necessary input file. */
2066 number_input_files = 1;
2067 input_file_name = xmalloc (sizeof *input_file_name);
2068 file_line_count = xmalloc (sizeof *file_line_count);
2069 text_buffers = xmalloc (sizeof *text_buffers);
2070 if (!*argv[optind] || STREQ (argv[optind], "-"))
2071 input_file_name[0] = NULL;
2072 else
2073 input_file_name[0] = argv[optind];
2074 optind++;
2076 /* Redirect standard output, only if requested. */
2078 if (optind < argc)
2080 if (! freopen (argv[optind], "w", stdout))
2081 die (EXIT_FAILURE, errno, "%s", quotef (argv[optind]));
2082 optind++;
2085 /* Diagnose any other argument as an error. */
2087 if (optind < argc)
2089 error (0, 0, _("extra operand %s"), quote (argv[optind]));
2090 usage (EXIT_FAILURE);
2094 /* If the output format has not been explicitly selected, choose dumb
2095 terminal format if GNU extensions are enabled, else 'roff' format. */
2097 if (output_format == UNKNOWN_FORMAT)
2098 output_format = gnu_extensions ? DUMB_FORMAT : ROFF_FORMAT;
2100 /* Initialize the main tables. */
2102 initialize_regex ();
2104 /* Read 'Break character' file, if any. */
2106 if (break_file)
2107 digest_break_file (break_file);
2109 /* Read 'Ignore words' file and 'Only words' files, if any. If any of
2110 these files is empty, reset the name of the file to NULL, to avoid
2111 unnecessary calls to search_table. */
2113 if (ignore_file)
2115 digest_word_file (ignore_file, &ignore_table);
2116 if (ignore_table.length == 0)
2117 ignore_file = NULL;
2120 if (only_file)
2122 digest_word_file (only_file, &only_table);
2123 if (only_table.length == 0)
2124 only_file = NULL;
2127 /* Prepare to study all the input files. */
2129 number_of_occurs[0] = 0;
2130 total_line_count = 0;
2131 maximum_word_length = 0;
2132 reference_max_width = 0;
2134 for (file_index = 0; file_index < number_input_files; file_index++)
2136 BLOCK *text_buffer = text_buffers + file_index;
2138 /* Read the file in core, then study it. */
2140 swallow_file_in_memory (input_file_name[file_index], text_buffer);
2141 find_occurs_in_text (file_index);
2143 /* Maintain for each file how many lines has been read so far when its
2144 end is reached. Incrementing the count first is a simple kludge to
2145 handle a possible incomplete line at end of file. */
2147 total_line_count++;
2148 file_line_count[file_index] = total_line_count;
2151 /* Do the output process phase. */
2153 sort_found_occurs ();
2154 fix_output_parameters ();
2155 generate_all_output ();
2157 /* All done. */
2159 return EXIT_SUCCESS;