#include <err.h>
#include <limits.h>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include "ctags.h"
NODE *head;
bool _wht[256], _itk[256], _btk[256];
FILE *inf;
FILE *outf;
long lineftell;
int lineno;
int dflag;
int vflag;
int wflag;
int xflag;
char *curfile;
char searchar = '/';
char lbuf[LINE_MAX];
void init(void);
void find_entries(char *);
void preload_entries(char *, int, char *[]);
int
main(int argc, char *argv[])
{
static char *outfile = "tags";
int aflag;
int uflag;
int exit_val;
int step;
int ch;
if (pledge("stdio rpath wpath cpath", NULL) == -1)
err(1, "pledge");
aflag = uflag = NO;
while ((ch = getopt(argc, argv, "BFadf:tuwvx")) != -1)
switch(ch) {
case 'B':
searchar = '?';
break;
case 'F':
searchar = '/';
break;
case 'a':
aflag = 1;
break;
case 'd':
dflag = 1;
break;
case 'f':
outfile = optarg;
break;
case 't':
break;
case 'u':
uflag = 1;
break;
case 'w':
wflag = 1;
break;
case 'v':
vflag = 1;
case 'x':
xflag = 1;
break;
default:
goto usage;
}
argv += optind;
argc -= optind;
if (!argc) {
usage: (void)fprintf(stderr,
"usage: ctags [-aBdFuvwx] [-f tagsfile] file ...\n");
exit(1);
}
init();
if (uflag && !vflag && !xflag)
preload_entries(outfile, argc, argv);
for (exit_val = step = 0; step < argc; ++step)
if (!(inf = fopen(argv[step], "r"))) {
warn("%s", argv[step]);
exit_val = 1;
}
else {
curfile = argv[step];
find_entries(argv[step]);
(void)fclose(inf);
}
if (head) {
if (xflag)
put_entries(head);
else {
if (!(outf = fopen(outfile, aflag ? "a" : "w")))
err(exit_val, "%s", outfile);
put_entries(head);
(void)fclose(outf);
}
}
exit(exit_val);
}
void
init(void)
{
int i;
unsigned char *sp;
for (i = 0; i < 256; i++)
_wht[i] = _itk[i] = _btk[i] = NO;
#define CWHITE " \f\t\n"
for (sp = CWHITE; *sp; sp++)
_wht[*sp] = YES;
#define CINTOK "ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz0123456789"
for (sp = CINTOK; *sp; sp++)
_itk[*sp] = YES;
#define CBEGIN "ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz"
for (sp = CBEGIN; *sp; sp++)
_btk[*sp] = YES;
}
void
find_entries(char *file)
{
char *cp;
lineno = 0;
if ((cp = strrchr(file, '.'))) {
if (cp[1] == 'l' && !cp[2]) {
int c;
for (;;) {
if (GETC(==, EOF))
return;
if (!iswhite(c)) {
rewind(inf);
break;
}
}
#define LISPCHR ";(["
if (strchr(LISPCHR, c)) {
l_entries();
return;
}
else {
toss_yysec();
(void)strlcpy(lbuf, "%%$", sizeof lbuf);
pfnote("yylex", lineno);
rewind(inf);
}
}
else if (cp[1] == 'y' && !cp[2]) {
toss_yysec();
(void)strlcpy(lbuf, "%%$", sizeof lbuf);
pfnote("yyparse", lineno);
y_entries();
}
else if ((cp[1] != 'c' && cp[1] != 'h') && !cp[2]) {
if (PF_funcs())
return;
rewind(inf);
}
}
c_entries();
}
void
preload_entries(char *tagsfile, int argc, char *argv[])
{
FILE *fp;
char line[LINE_MAX];
char *entry = NULL;
char *file = NULL;
char *pattern = NULL;
char *eol;
int i;
in_preload = YES;
if ((fp = fopen(tagsfile, "r")) == NULL)
err(1, "preload_entries: %s", tagsfile);
while (1) {
next:
if (fgets(line, sizeof(line), fp) == NULL)
break;
if ((eol = strchr(line, '\n')) == NULL)
errx(1, "preload_entries: line too long");
*eol = '\0';
entry = line;
if ((file = strchr(line, '\t')) == NULL)
errx(1, "preload_entries: couldn't parse entry: %s",
tagsfile);
*file = '\0';
file++;
if ((pattern = strchr(file, '\t')) == NULL)
errx(1, "preload_entries: couldn't parse filename: %s",
tagsfile);
*pattern = '\0';
for(i = 0; i < argc; i++)
if (strcmp(file, argv[i]) == 0)
goto next;
pattern++;
if ((pattern[0] == '/' || pattern[0] == '?')
&& pattern[1] == '^') {
i = strlen(pattern);
if (pattern[i-1] == pattern[0])
pattern[i-1] = '\0';
else
errx(1, "preload_entries: couldn't parse "
"pattern: %s", tagsfile);
pattern += 2;
}
if ((curfile = strdup(file)) == NULL)
err(1, "preload_entries: strdup");
(void)strlcpy(lbuf, pattern, sizeof(lbuf));
pfnote(entry, 0);
}
if (ferror(fp))
err(1, "preload_entries: fgets");
(void)fclose(fp);
in_preload = NO;
}