2023-07-27 14:04:50 -04:00
|
|
|
/*===========================================================================
|
2023-08-04 13:49:03 -04:00
|
|
|
Copyright (c) 1998-2000, The Santa Cruz Operation
|
2023-07-27 14:04:50 -04:00
|
|
|
All rights reserved.
|
2023-08-04 13:49:03 -04:00
|
|
|
|
2023-07-27 14:04:50 -04:00
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
*Redistributions of source code must retain the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
*Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
this list of conditions and the following disclaimer in the documentation
|
|
|
|
and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
*Neither name of The Santa Cruz Operation nor the names of its contributors
|
|
|
|
may be used to endorse or promote products derived from this software
|
2023-08-04 13:49:03 -04:00
|
|
|
without specific prior written permission.
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ``AS
|
2023-08-04 15:09:58 -04:00
|
|
|
IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT falseT LIMITED TO,
|
2023-07-27 14:04:50 -04:00
|
|
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
2023-08-04 15:19:25 -04:00
|
|
|
PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE
|
2023-07-27 14:04:50 -04:00
|
|
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
2023-08-04 15:09:58 -04:00
|
|
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT falseT LIMITED TO, PROCUREMENT OF
|
2023-07-27 14:04:50 -04:00
|
|
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
INTERRUPTION)
|
|
|
|
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
|
2023-08-04 13:49:03 -04:00
|
|
|
DAMAGE.
|
2023-07-27 14:04:50 -04:00
|
|
|
=========================================================================*/
|
|
|
|
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
/* cscope - interactive C symbol cross-reference
|
2023-07-27 14:04:50 -04:00
|
|
|
*
|
2023-08-04 13:49:03 -04:00
|
|
|
* main functions
|
2023-07-27 14:04:50 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "global.h"
|
|
|
|
|
|
|
|
#include "build.h"
|
|
|
|
#include "vp.h"
|
2023-08-04 13:49:03 -04:00
|
|
|
#include "version.h" /* FILEVERSION and FIXVERSION */
|
|
|
|
#include "scanner.h"
|
2023-07-27 14:04:50 -04:00
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
#include <stdlib.h> /* atoi */
|
2023-07-27 14:04:50 -04:00
|
|
|
#include <ncurses.h>
|
2023-08-04 13:49:03 -04:00
|
|
|
#include <sys/types.h> /* needed by stat.h */
|
|
|
|
#include <sys/stat.h> /* stat */
|
2023-07-27 14:04:50 -04:00
|
|
|
#include <signal.h>
|
|
|
|
#include <getopt.h>
|
|
|
|
|
|
|
|
/* defaults for unset environment variables */
|
2023-08-04 13:49:03 -04:00
|
|
|
#define EDITOR "vi"
|
|
|
|
#define HOME "/" /* no $HOME --> use root directory */
|
|
|
|
#define SHELL "sh"
|
|
|
|
#define LINEFLAG "+%s" /* default: used by vi and emacs */
|
|
|
|
#define TMPDIR "/tmp"
|
2023-07-27 14:04:50 -04:00
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
/* note: these digraph character frequencies were calculated from possible
|
2023-07-27 14:04:50 -04:00
|
|
|
printable digraphs in the cross-reference for the C compiler */
|
2023-08-04 13:49:03 -04:00
|
|
|
char dichar1[] = " teisaprnl(of)=c"; /* 16 most frequent first chars */
|
|
|
|
char dichar2[] = " tnerpla"; /* 8 most frequent second chars
|
|
|
|
using the above as first chars */
|
|
|
|
char dicode1[256]; /* digraph first character code */
|
|
|
|
char dicode2[256]; /* digraph second character code */
|
|
|
|
|
|
|
|
char *editor, *shell, *lineflag; /* environment variables */
|
|
|
|
char *home; /* Home directory */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool lineflagafterfile;
|
2023-08-04 13:49:03 -04:00
|
|
|
char *argv0; /* command name */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool compress = true; /* compress the characters in the crossref */
|
|
|
|
bool dbtruncated; /* database symbols are truncated to 8 chars */
|
2023-08-04 13:49:03 -04:00
|
|
|
int dispcomponents = 1; /* file path components to display */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool editallprompt = true; /* prompt between editing files */
|
2023-08-04 13:49:03 -04:00
|
|
|
unsigned int fileargc; /* file argument count */
|
|
|
|
char **fileargv; /* file argument values */
|
|
|
|
int fileversion; /* cross-reference file version */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool incurses = false; /* in curses */
|
|
|
|
bool invertedindex; /* the database has an inverted index */
|
|
|
|
bool isuptodate; /* consider the crossref up-to-date */
|
|
|
|
bool kernelmode; /* don't use DFLT_INCDIR - bad for kernels */
|
|
|
|
bool linemode = false; /* use line oriented user interface */
|
|
|
|
bool verbosemode = false; /* print extra information on line mode */
|
|
|
|
bool recurse_dir = false; /* recurse dirs when searching for src files */
|
2023-08-04 13:49:03 -04:00
|
|
|
char *namefile; /* file of file names */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool ogs = false; /* display OGS book and subsystem names */
|
2023-08-04 13:49:03 -04:00
|
|
|
char *prependpath; /* prepend path to file names */
|
|
|
|
FILE *refsfound; /* references found file */
|
|
|
|
char temp1[PATHLEN + 1]; /* temporary file name */
|
|
|
|
char temp2[PATHLEN + 1]; /* temporary file name */
|
|
|
|
char tempdirpv[PATHLEN + 1]; /* private temp directory */
|
|
|
|
long totalterms; /* total inverted index terms */
|
2023-08-04 15:09:58 -04:00
|
|
|
bool trun_syms; /* truncate symbols to 8 characters */
|
2023-08-04 13:49:03 -04:00
|
|
|
char tempstring[TEMPSTRING_LEN + 1]; /* use this as a buffer, instead of 'yytext',
|
|
|
|
* which had better be left alone */
|
|
|
|
char *tmpdir; /* temporary directory */
|
|
|
|
|
|
|
|
static char path[PATHLEN + 1]; /* file path */
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
/* Internal prototypes: */
|
2023-08-04 13:49:03 -04:00
|
|
|
static void skiplist(FILE *oldrefs);
|
|
|
|
static void initcompress(void);
|
|
|
|
static inline void readenv(void);
|
|
|
|
static inline void linemode_event_loop(void);
|
|
|
|
static inline void screenmode_event_loop(void);
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
#if defined(KEY_RESIZE) && !defined(__DJGPP__)
|
2023-08-04 13:49:03 -04:00
|
|
|
void
|
2023-07-27 14:04:50 -04:00
|
|
|
sigwinch_handler(int sig, siginfo_t *info, void *unused)
|
|
|
|
{
|
2023-08-09 07:49:11 -04:00
|
|
|
UNUSED(sig);
|
|
|
|
UNUSED(info);
|
|
|
|
UNUSED(unused);
|
|
|
|
|
2023-08-04 15:09:58 -04:00
|
|
|
if(incurses == true){
|
2023-07-27 14:04:50 -04:00
|
|
|
ungetch(KEY_RESIZE);
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-07-29 09:40:48 -04:00
|
|
|
static
|
|
|
|
inline
|
|
|
|
void
|
|
|
|
siginit(void){
|
|
|
|
/* if running in the foreground */
|
|
|
|
if (signal(SIGINT, SIG_IGN) != SIG_IGN) {
|
2023-08-05 08:09:42 -04:00
|
|
|
/* cleanup on the interrupt and quit signals */
|
|
|
|
signal(SIGINT, myexit);
|
|
|
|
signal(SIGQUIT, myexit);
|
2023-07-29 09:40:48 -04:00
|
|
|
}
|
|
|
|
/* cleanup on the hangup signal */
|
|
|
|
signal(SIGHUP, myexit);
|
|
|
|
|
|
|
|
/* ditto the TERM signal */
|
|
|
|
signal(SIGTERM, myexit);
|
|
|
|
|
|
|
|
/* ignore PIPE signal, so myexit() will have a chance to clean up in
|
|
|
|
* linemode, while in curses mode the "|" command can cause a pipe signal
|
|
|
|
* too
|
|
|
|
*/
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
2023-08-04 15:09:58 -04:00
|
|
|
if (linemode == false) {
|
2023-08-05 08:09:42 -04:00
|
|
|
signal(SIGWINCH, sigwinch_handler);
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
2023-07-29 09:40:48 -04:00
|
|
|
}
|
|
|
|
|
2023-08-04 13:12:06 -04:00
|
|
|
void
|
2023-08-11 13:20:52 -04:00
|
|
|
cannotopen(const char *const file)
|
2023-08-04 13:12:06 -04:00
|
|
|
{
|
|
|
|
posterr("Cannot open file %s", file);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME MTE - should use postfatal here */
|
|
|
|
void
|
2023-08-11 13:20:52 -04:00
|
|
|
cannotwrite(const char *const file)
|
2023-08-04 13:12:06 -04:00
|
|
|
{
|
2023-08-04 13:49:03 -04:00
|
|
|
char msg[MSGLEN + 1];
|
2023-08-04 13:12:06 -04:00
|
|
|
|
|
|
|
snprintf(msg, sizeof(msg), "Removed file %s because write failed", file);
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
myperror(msg); /* display the reason */
|
2023-08-04 13:12:06 -04:00
|
|
|
|
|
|
|
unlink(file);
|
2023-08-04 13:49:03 -04:00
|
|
|
myexit(1); /* calls exit(2), which closes files */
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* set up the digraph character tables for text compression */
|
|
|
|
static void
|
|
|
|
initcompress(void)
|
|
|
|
{
|
2023-08-04 13:49:03 -04:00
|
|
|
int i;
|
|
|
|
|
2023-08-04 15:09:58 -04:00
|
|
|
if (compress == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
for (i = 0; i < 16; ++i) {
|
|
|
|
dicode1[(unsigned char) (dichar1[i])] = i * 8 + 1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < 8; ++i) {
|
|
|
|
dicode2[(unsigned char) (dichar2[i])] = i + 1;
|
|
|
|
}
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* skip the list in the cross-reference file */
|
|
|
|
|
|
|
|
static void
|
|
|
|
skiplist(FILE *oldrefs)
|
|
|
|
{
|
2023-08-04 13:49:03 -04:00
|
|
|
int i;
|
|
|
|
|
2023-08-04 13:12:06 -04:00
|
|
|
if (fscanf(oldrefs, "%d", &i) != 1) {
|
2023-08-11 14:27:20 -04:00
|
|
|
postfatal(PROGRAM_NAME ": cannot read list size from file %s\n", reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
while (--i >= 0) {
|
2023-08-04 13:49:03 -04:00
|
|
|
if (fscanf(oldrefs, "%*s") != 0) {
|
2023-08-11 14:27:20 -04:00
|
|
|
postfatal(PROGRAM_NAME ": cannot read list name from file %s\n", reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* cleanup and exit */
|
|
|
|
void
|
|
|
|
myexit(int sig)
|
|
|
|
{
|
2023-08-04 13:49:03 -04:00
|
|
|
/* Close file before unlinking it. DOS absolutely needs it */
|
|
|
|
if (refsfound != NULL){
|
|
|
|
fclose(refsfound);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* remove any temporary files */
|
|
|
|
if (temp1[0] != '\0') {
|
|
|
|
unlink(temp1);
|
|
|
|
unlink(temp2);
|
|
|
|
rmdir(tempdirpv);
|
|
|
|
}
|
|
|
|
/* restore the terminal to its original mode */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (incurses == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
exitcurses();
|
|
|
|
}
|
|
|
|
/* dump core for debugging on the quit signal */
|
|
|
|
if (sig == SIGQUIT) {
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
/* HBB 20000421: be nice: free allocated data */
|
|
|
|
freefilelist();
|
|
|
|
freeinclist();
|
|
|
|
freesrclist();
|
|
|
|
freecrossref();
|
|
|
|
free_newbuildfiles();
|
|
|
|
|
2023-08-04 15:09:58 -04:00
|
|
|
if( remove_symfile_onexit == true ) {
|
2023-08-04 13:49:03 -04:00
|
|
|
unlink( reffile );
|
|
|
|
unlink( invname );
|
|
|
|
unlink( invpost );
|
|
|
|
}
|
|
|
|
|
|
|
|
exit(sig);
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void readenv(void){
|
|
|
|
editor = mygetenv("EDITOR", EDITOR);
|
|
|
|
editor = mygetenv("VIEWER", editor); /* use viewer if set */
|
2023-08-04 13:49:03 -04:00
|
|
|
editor = mygetenv("CSCOPE_EDITOR", editor); /* has last word */
|
2023-08-04 13:12:06 -04:00
|
|
|
home = mygetenv("HOME", HOME);
|
|
|
|
shell = mygetenv("SHELL", SHELL);
|
|
|
|
lineflag = mygetenv("CSCOPE_LINEFLAG", LINEFLAG);
|
|
|
|
lineflagafterfile = getenv("CSCOPE_LINEFLAG_AFTER_FILE") ? 1 : 0;
|
|
|
|
tmpdir = mygetenv("TMPDIR", TMPDIR);
|
|
|
|
}
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
static inline void linemode_event_loop(void){
|
|
|
|
int c;
|
|
|
|
|
|
|
|
if (*input_line != '\0') { /* do any optional search */
|
2023-08-10 15:06:17 -04:00
|
|
|
if (search(input_line) == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
/* print the total number of lines in
|
|
|
|
* verbose mode */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (verbosemode == true)
|
2023-08-11 14:27:20 -04:00
|
|
|
printf(PROGRAM_NAME ": %d lines\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
totallines);
|
|
|
|
|
|
|
|
while ((c = getc(refsfound)) != EOF)
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
}
|
2023-08-04 15:09:58 -04:00
|
|
|
if (onesearch == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
myexit(0);
|
2023-08-10 15:06:17 -04:00
|
|
|
/* NOTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
for (char *s;;) {
|
|
|
|
char buf[PATLEN + 2];
|
|
|
|
|
|
|
|
printf(">> ");
|
|
|
|
fflush(stdout);
|
|
|
|
if (fgets(buf, sizeof(buf), stdin) == NULL) {
|
2023-08-10 15:06:17 -04:00
|
|
|
myexit(0);
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
/* remove any trailing newline character */
|
|
|
|
if (*(s = buf + strlen(buf) - 1) == '\n') {
|
|
|
|
*s = '\0';
|
|
|
|
}
|
|
|
|
switch (*buf) {
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
case '5':
|
|
|
|
case '6':
|
|
|
|
case '7':
|
|
|
|
case '8':
|
|
|
|
case '9': /* samuel only */
|
|
|
|
field = *buf - '0';
|
|
|
|
strcpy(input_line, buf + 1);
|
2023-08-10 15:06:17 -04:00
|
|
|
if (search(input_line) == false) {
|
2023-08-04 13:49:03 -04:00
|
|
|
printf("Unable to search database\n");
|
|
|
|
} else {
|
|
|
|
printf("cscope: %d lines\n", totallines);
|
|
|
|
while ((c = getc(refsfound)) != EOF) {
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'c': /* toggle caseless mode */
|
|
|
|
case ctrl('C'):
|
2023-08-04 15:09:58 -04:00
|
|
|
if (caseless == false) {
|
|
|
|
caseless = true;
|
2023-08-04 13:49:03 -04:00
|
|
|
} else {
|
2023-08-04 15:09:58 -04:00
|
|
|
caseless = false;
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
egrepcaseless(caseless);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'r': /* rebuild database cscope style */
|
|
|
|
case ctrl('R'):
|
|
|
|
freefilelist();
|
|
|
|
makefilelist();
|
|
|
|
/* FALLTHROUGH */
|
|
|
|
|
|
|
|
case 'R': /* rebuild database samuel style */
|
|
|
|
rebuild();
|
|
|
|
putchar('\n');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'C': /* clear file names */
|
|
|
|
freefilelist();
|
|
|
|
putchar('\n');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'F': /* add a file name */
|
|
|
|
strcpy(path, buf + 1);
|
2023-08-04 15:09:58 -04:00
|
|
|
if (infilelist(path) == false &&
|
2023-08-04 13:49:03 -04:00
|
|
|
(s = inviewpath(path)) != NULL) {
|
|
|
|
addsrcfile(s);
|
|
|
|
}
|
|
|
|
putchar('\n');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'q': /* quit */
|
|
|
|
case ctrl('D'):
|
|
|
|
case ctrl('Z'):
|
2023-08-10 15:06:17 -04:00
|
|
|
myexit(0);
|
|
|
|
/* NOTREACHED */
|
|
|
|
break;
|
2023-08-04 13:49:03 -04:00
|
|
|
default:
|
2023-08-11 14:27:20 -04:00
|
|
|
fprintf(stderr, PROGRAM_NAME ": unknown command '%s'\n", buf);
|
2023-08-04 13:49:03 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
static inline void screenmode_event_loop(void){
|
2023-08-04 13:12:06 -04:00
|
|
|
for (;;) {
|
2023-08-04 13:49:03 -04:00
|
|
|
display();
|
2023-08-08 14:34:15 -04:00
|
|
|
handle_input(wgetch(stdscr)); // NOTE: getch() does not return key codes
|
2023-08-04 13:12:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-27 14:04:50 -04:00
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
2023-08-04 13:49:03 -04:00
|
|
|
FILE *names; /* name file pointer */
|
|
|
|
int oldnum; /* number in old cross-ref */
|
|
|
|
FILE *oldrefs; /* old cross-reference file */
|
2023-07-27 14:04:50 -04:00
|
|
|
char *s;
|
|
|
|
unsigned int i;
|
|
|
|
pid_t pid;
|
2023-08-04 13:49:03 -04:00
|
|
|
struct stat stat_buf;
|
2023-07-27 14:04:50 -04:00
|
|
|
mode_t orig_umask;
|
2023-08-04 13:49:03 -04:00
|
|
|
|
2023-07-27 14:04:50 -04:00
|
|
|
yyin = stdin;
|
|
|
|
yyout = stdout;
|
|
|
|
/* save the command name for messages */
|
|
|
|
argv0 = argv[0];
|
|
|
|
|
|
|
|
/* set the options */
|
2023-08-04 13:49:03 -04:00
|
|
|
argv = parse_options(&argc, argv);
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
/* read the environment */
|
2023-08-04 13:49:03 -04:00
|
|
|
readenv();
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
/* XXX remove if/when clearerr() in dir.c does the right thing. */
|
|
|
|
if (namefile && strcmp(namefile, "-") == 0 && !buildonly) {
|
2023-08-11 14:27:20 -04:00
|
|
|
postfatal(PROGRAM_NAME ": Must use -b if file list comes from stdin\n");
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* make sure that tmpdir exists */
|
|
|
|
if (lstat (tmpdir, &stat_buf)) {
|
2023-08-11 14:27:20 -04:00
|
|
|
fprintf (stderr, PROGRAM_NAME ": Temporary directory %s does not exist or cannot be accessed\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
tmpdir);
|
|
|
|
fprintf (stderr,
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": Please create the directory or set the environment variable\n"
|
|
|
|
PROGRAM_NAME ": TMPDIR to a valid directory\n");
|
2023-08-04 13:49:03 -04:00
|
|
|
myexit(1);
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* create the temporary file names */
|
|
|
|
orig_umask = umask(S_IRWXG|S_IRWXO);
|
|
|
|
pid = getpid();
|
2023-08-11 14:27:20 -04:00
|
|
|
snprintf(tempdirpv, sizeof(tempdirpv), "%s/"PROGRAM_NAME".%d", tmpdir, pid);
|
2023-07-27 14:04:50 -04:00
|
|
|
if(mkdir(tempdirpv,S_IRWXU)) {
|
2023-08-04 13:49:03 -04:00
|
|
|
fprintf(stderr,
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": Could not create private temp dir %s\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
tempdirpv);
|
|
|
|
myexit(1);
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
umask(orig_umask);
|
|
|
|
|
2023-08-11 14:27:20 -04:00
|
|
|
snprintf(temp1, sizeof(temp1), "%s/"PROGRAM_NAME".1", tempdirpv);
|
|
|
|
snprintf(temp2, sizeof(temp2), "%s/"PROGRAM_NAME".2", tempdirpv);
|
2023-07-27 14:04:50 -04:00
|
|
|
|
|
|
|
/* if the database path is relative and it can't be created */
|
|
|
|
if (reffile[0] != '/' && access(".", WRITE) != 0) {
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
/* put it in the home directory if the database may not be
|
|
|
|
* up-to-date or doesn't exist in the relative directory,
|
|
|
|
* so a database in the current directory will be
|
|
|
|
* used instead of failing to open a non-existant database in
|
|
|
|
* the home directory
|
|
|
|
*/
|
|
|
|
snprintf(path, sizeof(path), "%s/%s", home, reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
if (isuptodate == false || access(path, READ) == 0) {
|
2023-08-04 13:49:03 -04:00
|
|
|
reffile = strdup(path);
|
|
|
|
snprintf(path, sizeof(path), "%s/%s", home, invname);
|
|
|
|
invname = strdup(path);
|
|
|
|
snprintf(path, sizeof(path), "%s/%s", home, invpost);
|
|
|
|
invpost = strdup(path);
|
|
|
|
}
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
siginit();
|
2023-07-27 14:04:50 -04:00
|
|
|
|
2023-08-04 15:09:58 -04:00
|
|
|
if (linemode == false) {
|
2023-08-04 13:49:03 -04:00
|
|
|
dispinit(); /* initialize display parameters */
|
2023-08-09 07:49:11 -04:00
|
|
|
postmsg(""); /* clear any build progress message */
|
2023-08-04 13:49:03 -04:00
|
|
|
display(); /* display the version number and input fields */
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* if the cross-reference is to be considered up-to-date */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (isuptodate == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
if ((oldrefs = vpfopen(reffile, "rb")) == NULL) {
|
2023-08-11 14:27:20 -04:00
|
|
|
postfatal(PROGRAM_NAME ": cannot open file %s\n", reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
/* get the crossref file version but skip the current directory */
|
2023-08-11 14:27:20 -04:00
|
|
|
if (fscanf(oldrefs, PROGRAM_NAME " %d %*s", &fileversion) != 1) {
|
|
|
|
postfatal(PROGRAM_NAME ": cannot read file version from file %s\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
if (fileversion >= 8) {
|
|
|
|
|
|
|
|
/* override these command line options */
|
2023-08-04 15:09:58 -04:00
|
|
|
compress = true;
|
|
|
|
invertedindex = false;
|
2023-08-04 13:49:03 -04:00
|
|
|
|
|
|
|
/* see if there are options in the database */
|
|
|
|
for (int c;;) {
|
|
|
|
getc(oldrefs); /* skip the blank */
|
|
|
|
if ((c = getc(oldrefs)) != '-') {
|
|
|
|
ungetc(c, oldrefs);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (getc(oldrefs)) {
|
|
|
|
case 'c': /* ASCII characters only */
|
2023-08-04 15:09:58 -04:00
|
|
|
compress = false;
|
2023-08-04 13:49:03 -04:00
|
|
|
break;
|
|
|
|
case 'q': /* quick search */
|
2023-08-04 15:09:58 -04:00
|
|
|
invertedindex = true;
|
2023-08-04 13:49:03 -04:00
|
|
|
fscanf(oldrefs, "%ld", &totalterms);
|
|
|
|
break;
|
|
|
|
case 'T': /* truncate symbols to 8 characters */
|
2023-08-04 15:09:58 -04:00
|
|
|
dbtruncated = true;
|
|
|
|
trun_syms = true;
|
2023-08-04 13:49:03 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
initcompress();
|
|
|
|
seek_to_trailer(oldrefs);
|
|
|
|
}
|
|
|
|
/* skip the source and include directory lists */
|
|
|
|
skiplist(oldrefs);
|
|
|
|
skiplist(oldrefs);
|
|
|
|
|
|
|
|
/* get the number of source files */
|
|
|
|
if (fscanf(oldrefs, "%lu", &nsrcfiles) != 1) {
|
|
|
|
postfatal(
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": cannot read source file size from file %s\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
/* get the source file list */
|
|
|
|
srcfiles = malloc(nsrcfiles * sizeof(*srcfiles));
|
|
|
|
if (fileversion >= 9) {
|
|
|
|
|
|
|
|
/* allocate the string space */
|
|
|
|
if (fscanf(oldrefs, "%d", &oldnum) != 1) {
|
|
|
|
postfatal(
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": cannot read string space size from file %s\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
s = malloc(oldnum);
|
|
|
|
getc(oldrefs); /* skip the newline */
|
|
|
|
|
|
|
|
/* read the strings */
|
|
|
|
if (fread(s, oldnum, 1, oldrefs) != 1) {
|
|
|
|
postfatal(
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": cannot read source file names from file %s\n",
|
2023-08-04 13:49:03 -04:00
|
|
|
reffile);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
/* change newlines to nulls */
|
|
|
|
for (i = 0; i < nsrcfiles; ++i) {
|
|
|
|
srcfiles[i] = s;
|
|
|
|
for (++s; *s != '\n'; ++s) {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
*s = '\0';
|
|
|
|
++s;
|
|
|
|
}
|
|
|
|
/* if there is a file of source file names */
|
|
|
|
if ((namefile != NULL && (names = vpfopen(namefile, "r")) != NULL)
|
|
|
|
|| (names = vpfopen(NAMEFILE, "r")) != NULL) {
|
|
|
|
|
|
|
|
/* read any -p option from it */
|
|
|
|
while (fgets(path, sizeof(path), names) != NULL && *path == '-') {
|
|
|
|
i = path[1];
|
|
|
|
s = path + 2; /* for "-Ipath" */
|
|
|
|
if (*s == '\0') { /* if "-I path" */
|
|
|
|
fgets(path, sizeof(path), names);
|
|
|
|
s = path;
|
|
|
|
}
|
|
|
|
switch (i) {
|
|
|
|
case 'p': /* file path components to display */
|
|
|
|
if (*s < '0' || *s > '9') {
|
2023-08-11 14:27:20 -04:00
|
|
|
posterr(PROGRAM_NAME ": -p option in file %s: missing or invalid numeric value\n", namefile);
|
2023-08-04 13:49:03 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
dispcomponents = atoi(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(names);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < nsrcfiles; ++i) {
|
|
|
|
if (!fgets(path, sizeof(path), oldrefs) ) {
|
|
|
|
postfatal(
|
2023-08-11 14:27:20 -04:00
|
|
|
PROGRAM_NAME ": cannot read source file name from file %s\n",
|
|
|
|
reffile
|
|
|
|
);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
srcfiles[i] = strdup(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(oldrefs);
|
2023-07-27 14:04:50 -04:00
|
|
|
} else {
|
2023-08-04 13:49:03 -04:00
|
|
|
/* save the file arguments */
|
|
|
|
fileargc = argc;
|
|
|
|
fileargv = argv;
|
|
|
|
|
|
|
|
/* get source directories from the environment */
|
|
|
|
if ((s = getenv("SOURCEDIRS")) != NULL) {
|
|
|
|
sourcedir(s);
|
|
|
|
}
|
|
|
|
/* make the source file list */
|
|
|
|
srcfiles = malloc(msrcfiles * sizeof(*srcfiles));
|
|
|
|
makefilelist();
|
|
|
|
if (nsrcfiles == 0) {
|
2023-08-11 14:27:20 -04:00
|
|
|
postfatal(PROGRAM_NAME ": no source files found\n");
|
|
|
|
/* NOTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
|
|
|
/* get include directories from the environment */
|
|
|
|
if ((s = getenv("INCLUDEDIRS")) != NULL) {
|
|
|
|
includedir(s);
|
|
|
|
}
|
|
|
|
/* add /usr/include to the #include directory list,
|
|
|
|
but not in kernelmode... kernels tend not to use it. */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (kernelmode == false) {
|
2023-08-04 13:49:03 -04:00
|
|
|
if (NULL != (s = getenv("INCDIR"))) {
|
|
|
|
includedir(s);
|
|
|
|
} else {
|
|
|
|
includedir(DFLT_INCDIR);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* initialize the C keyword table */
|
|
|
|
initsymtab();
|
|
|
|
|
|
|
|
/* Tell build.c about the filenames to create: */
|
|
|
|
setup_build_filenames(reffile);
|
|
|
|
|
|
|
|
/* build the cross-reference */
|
|
|
|
initcompress();
|
2023-08-04 15:09:58 -04:00
|
|
|
if (linemode == false || verbosemode == true) { /* display if verbose as well */
|
2023-08-04 13:49:03 -04:00
|
|
|
postmsg("Building cross-reference...");
|
|
|
|
}
|
|
|
|
build();
|
2023-08-04 15:09:58 -04:00
|
|
|
if (linemode == false ) {
|
2023-08-09 07:49:11 -04:00
|
|
|
postmsg(""); /* clear any build progress message */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
2023-08-04 15:09:58 -04:00
|
|
|
if (buildonly == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
myexit(0);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
}
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
opendatabase();
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
/* if using the line oriented user interface so cscope can be a
|
2023-07-27 14:04:50 -04:00
|
|
|
subprocess to emacs or samuel */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (linemode == true) {
|
2023-08-04 13:49:03 -04:00
|
|
|
linemode_event_loop();
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
/* pause before clearing the screen if there have been error messages */
|
2023-08-04 15:09:58 -04:00
|
|
|
if (errorsfound == true) {
|
|
|
|
errorsfound = false;
|
2023-08-04 13:49:03 -04:00
|
|
|
askforreturn();
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
/* do any optional search */
|
2023-08-04 13:12:06 -04:00
|
|
|
if (*input_line != '\0') {
|
2023-08-04 13:49:03 -04:00
|
|
|
//command(ctrl('Y')); /* search */ // XXX: fix
|
2023-07-27 14:04:50 -04:00
|
|
|
} else if (reflines != NULL) {
|
2023-08-04 13:49:03 -04:00
|
|
|
/* read any symbol reference lines file */
|
|
|
|
readrefs(reflines);
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|
|
|
|
|
2023-08-04 13:49:03 -04:00
|
|
|
screenmode_event_loop();
|
2023-07-27 14:04:50 -04:00
|
|
|
/* cleanup and exit */
|
|
|
|
myexit(0);
|
2023-08-04 15:09:58 -04:00
|
|
|
/* falseTREACHED */
|
2023-08-04 13:49:03 -04:00
|
|
|
return 0; /* avoid warning... */
|
2023-07-27 14:04:50 -04:00
|
|
|
}
|