busybox/grep.c

196 lines
3.2 KiB
C
Raw Normal View History

1999-10-06 00:24:54 +08:00
/*
* Copyright (c) 1999 by David I. Bell
* Permission is granted to use, distribute, or modify this source,
* provided that this copyright notice remains intact.
*
* The "grep" command, taken from sash.
* This provides basic file searching.
*
* Permission to distribute this code under the GPL has been granted.
* Modified for busybox by Erik Andersen <andersee@debian.org> <andersen@lineo.com>
*/
#include "internal.h"
#include <stdio.h>
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <signal.h>
#include <time.h>
#include <ctype.h>
const char grep_usage[] =
1999-10-07 16:30:23 +08:00
"Search the input file(s) for lines matching the given pattern.\n"
"\tI search stdin if no files are given.\n"
"\tI can't grok full regular expressions.\n"
"usage: grep [in] PATTERN [FILES]...\n"
"\ti=ignore case, n=list line numbers\n";
1999-10-06 00:24:54 +08:00
1999-10-13 06:26:06 +08:00
/*
* See if the specified word is found in the specified string.
*/
static int search (const char *string, const char *word, int ignoreCase)
{
const char *cp1;
const char *cp2;
int len;
int lowFirst;
int ch1;
int ch2;
len = strlen (word);
if (!ignoreCase) {
while (TRUE) {
string = strchr (string, word[0]);
if (string == NULL)
return FALSE;
if (memcmp (string, word, len) == 0)
return TRUE;
string++;
}
}
/*
* Here if we need to check case independence.
* Do the search by lower casing both strings.
*/
lowFirst = *word;
if (isupper (lowFirst))
lowFirst = tolower (lowFirst);
while (TRUE) {
while (*string && (*string != lowFirst) &&
(!isupper (*string) || (tolower (*string) != lowFirst))) {
string++;
}
if (*string == '\0')
return FALSE;
cp1 = string;
cp2 = word;
do {
if (*cp2 == '\0')
return TRUE;
ch1 = *cp1++;
if (isupper (ch1))
ch1 = tolower (ch1);
ch2 = *cp2++;
if (isupper (ch2))
ch2 = tolower (ch2);
}
while (ch1 == ch2);
string++;
}
return (TRUE);
}
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
extern int grep_main (int argc, char **argv)
1999-10-06 00:24:54 +08:00
{
1999-10-07 16:30:23 +08:00
FILE *fp;
const char *word;
const char *name;
const char *cp;
1999-10-09 08:25:00 +08:00
int tellName;
int ignoreCase;
int tellLine;
1999-10-07 16:30:23 +08:00
long line;
char buf[BUF_SIZE];
ignoreCase = FALSE;
tellLine = FALSE;
argc--;
argv++;
if (argc < 1) {
fprintf (stderr, "%s", grep_usage);
return 1;
}
if (**argv == '-') {
1999-10-06 00:24:54 +08:00
argc--;
1999-10-07 16:30:23 +08:00
cp = *argv++;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
while (*++cp)
switch (*cp) {
case 'i':
ignoreCase = TRUE;
break;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
case 'n':
tellLine = TRUE;
break;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
default:
fprintf (stderr, "Unknown option\n");
return 1;
}
}
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
word = *argv++;
argc--;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
tellName = (argc > 1);
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
while (argc-- > 0) {
name = *argv++;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
fp = fopen (name, "r");
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
if (fp == NULL) {
perror (name);
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
continue;
}
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
line = 0;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
while (fgets (buf, sizeof (buf), fp)) {
line++;
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
cp = &buf[strlen (buf) - 1];
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
if (*cp != '\n')
fprintf (stderr, "%s: Line too long\n", name);
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
if (search (buf, word, ignoreCase)) {
if (tellName)
printf ("%s: ", name);
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
if (tellLine)
printf ("%ld: ", line);
1999-10-06 00:24:54 +08:00
1999-10-07 16:30:23 +08:00
fputs (buf, stdout);
}
1999-10-06 00:24:54 +08:00
}
1999-10-07 16:30:23 +08:00
if (ferror (fp))
perror (name);
fclose (fp);
}
return 0;
1999-10-06 00:24:54 +08:00
}
1999-10-13 06:26:06 +08:00
/* END CODE */
1999-10-07 16:30:23 +08:00
1999-10-06 00:24:54 +08:00