Remove extra parenthesis.
[dragonfly.git] / contrib / nvi / ex / ex_cscope.c
blobc2fa0a5c454be0998e7f1539d9978bd7230c3eef
1 /*-
2 * Copyright (c) 1994, 1996
3 * Rob Mayoff. All rights reserved.
4 * Copyright (c) 1996
5 * Keith Bostic. All rights reserved.
7 * See the LICENSE file for redistribution information.
8 */
10 #include "config.h"
12 #ifndef lint
13 static const char sccsid[] = "@(#)ex_cscope.c 10.13 (Berkeley) 9/15/96";
14 #endif /* not lint */
16 #include <sys/param.h>
17 #include <sys/types.h> /* XXX: param.h may not have included types.h */
18 #include <sys/queue.h>
19 #include <sys/stat.h>
20 #include <sys/time.h>
21 #include <sys/wait.h>
23 #include <bitstring.h>
24 #include <ctype.h>
25 #include <errno.h>
26 #include <fcntl.h>
27 #include <limits.h>
28 #include <stddef.h>
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <string.h>
32 #include <termios.h>
33 #include <unistd.h>
35 #include "../common/common.h"
36 #include "pathnames.h"
37 #include "tag.h"
39 #define CSCOPE_DBFILE "cscope.out"
40 #define CSCOPE_PATHS "cscope.tpath"
43 * 0name find all uses of name
44 * 1name find definition of name
45 * 2name find all function calls made from name
46 * 3name find callers of name
47 * 4string find text string (cscope 12.9)
48 * 4name find assignments to name (cscope 13.3)
49 * 5pattern change pattern -- NOT USED
50 * 6pattern find pattern
51 * 7name find files with name as substring
52 * 8name find files #including name
54 #define FINDHELP "\
55 find c|d|e|f|g|i|s|t buffer|pattern\n\
56 c: find callers of name\n\
57 d: find all function calls made from name\n\
58 e: find pattern\n\
59 f: find files with name as substring\n\
60 g: find definition of name\n\
61 i: find files #including name\n\
62 s: find all uses of name\n\
63 t: find assignments to name"
65 static int cscope_add __P((SCR *, EXCMD *, char *));
66 static int cscope_find __P((SCR *, EXCMD*, char *));
67 static int cscope_help __P((SCR *, EXCMD *, char *));
68 static int cscope_kill __P((SCR *, EXCMD *, char *));
69 static int cscope_reset __P((SCR *, EXCMD *, char *));
71 typedef struct _cc {
72 char *name;
73 int (*function) __P((SCR *, EXCMD *, char *));
74 char *help_msg;
75 char *usage_msg;
76 } CC;
78 static CC const cscope_cmds[] = {
79 { "add", cscope_add,
80 "Add a new cscope database", "add file | directory" },
81 { "find", cscope_find,
82 "Query the databases for a pattern", FINDHELP },
83 { "help", cscope_help,
84 "Show help for cscope commands", "help [command]" },
85 { "kill", cscope_kill,
86 "Kill a cscope connection", "kill number" },
87 { "reset", cscope_reset,
88 "Discard all current cscope connections", "reset" },
89 { NULL }
92 static TAGQ *create_cs_cmd __P((SCR *, char *, size_t *));
93 static int csc_help __P((SCR *, char *));
94 static void csc_file __P((SCR *,
95 CSC *, char *, char **, size_t *, int *));
96 static int get_paths __P((SCR *, CSC *));
97 static CC const *lookup_ccmd __P((char *));
98 static int parse __P((SCR *, CSC *, TAGQ *, int *));
99 static int read_prompt __P((SCR *, CSC *));
100 static int run_cscope __P((SCR *, CSC *, char *));
101 static int start_cscopes __P((SCR *, EXCMD *));
102 static int terminate __P((SCR *, CSC *, int));
105 * ex_cscope --
106 * Perform an ex cscope.
108 * PUBLIC: int ex_cscope __P((SCR *, EXCMD *));
111 ex_cscope(sp, cmdp)
112 SCR *sp;
113 EXCMD *cmdp;
115 CC const *ccp;
116 EX_PRIVATE *exp;
117 int i;
118 char *cmd, *p;
120 /* Initialize the default cscope directories. */
121 exp = EXP(sp);
122 if (!F_ISSET(exp, EXP_CSCINIT) && start_cscopes(sp, cmdp))
123 return (1);
124 F_SET(exp, EXP_CSCINIT);
126 /* Skip leading whitespace. */
127 for (p = cmdp->argv[0]->bp, i = cmdp->argv[0]->len; i > 0; --i, ++p)
128 if (!isspace(*p))
129 break;
130 if (i == 0)
131 goto usage;
133 /* Skip the command to any arguments. */
134 for (cmd = p; i > 0; --i, ++p)
135 if (isspace(*p))
136 break;
137 if (*p != '\0') {
138 *p++ = '\0';
139 for (; *p && isspace(*p); ++p);
142 if ((ccp = lookup_ccmd(cmd)) == NULL) {
143 usage: msgq(sp, M_ERR, "309|Use \"cscope help\" for help");
144 return (1);
147 /* Call the underlying function. */
148 return (ccp->function(sp, cmdp, p));
152 * start_cscopes --
153 * Initialize the cscope package.
155 static int
156 start_cscopes(sp, cmdp)
157 SCR *sp;
158 EXCMD *cmdp;
160 size_t blen, len;
161 char *bp, *cscopes, *p, *t;
164 * EXTENSION #1:
166 * If the CSCOPE_DIRS environment variable is set, we treat it as a
167 * list of cscope directories that we're using, similar to the tags
168 * edit option.
170 * XXX
171 * This should probably be an edit option, although that implies that
172 * we start/stop cscope processes periodically, instead of once when
173 * the editor starts.
175 if ((cscopes = getenv("CSCOPE_DIRS")) == NULL)
176 return (0);
177 len = strlen(cscopes);
178 GET_SPACE_RET(sp, bp, blen, len);
179 memcpy(bp, cscopes, len + 1);
181 for (cscopes = t = bp; (p = strsep(&t, "\t :")) != NULL;)
182 if (*p != '\0')
183 (void)cscope_add(sp, cmdp, p);
185 FREE_SPACE(sp, bp, blen);
186 return (0);
190 * cscope_add --
191 * The cscope add command.
193 static int
194 cscope_add(sp, cmdp, dname)
195 SCR *sp;
196 EXCMD *cmdp;
197 char *dname;
199 struct stat sb;
200 EX_PRIVATE *exp;
201 CSC *csc;
202 size_t len;
203 int cur_argc;
204 char *dbname, path[MAXPATHLEN];
206 exp = EXP(sp);
209 * 0 additional args: usage.
210 * 1 additional args: matched a file.
211 * >1 additional args: object, too many args.
213 cur_argc = cmdp->argc;
214 if (argv_exp2(sp, cmdp, dname, strlen(dname)))
215 return (1);
216 if (cmdp->argc == cur_argc) {
217 (void)csc_help(sp, "add");
218 return (1);
220 if (cmdp->argc == cur_argc + 1)
221 dname = cmdp->argv[cur_argc]->bp;
222 else {
223 ex_emsg(sp, dname, EXM_FILECOUNT);
224 return (1);
228 * The user can specify a specific file (so they can have multiple
229 * Cscope databases in a single directory) or a directory. If the
230 * file doesn't exist, we're done. If it's a directory, append the
231 * standard database file name and try again. Store the directory
232 * name regardless so that we can use it as a base for searches.
234 if (stat(dname, &sb)) {
235 msgq(sp, M_SYSERR, dname);
236 return (1);
238 if (S_ISDIR(sb.st_mode)) {
239 (void)snprintf(path, sizeof(path),
240 "%s/%s", dname, CSCOPE_DBFILE);
241 if (stat(path, &sb)) {
242 msgq(sp, M_SYSERR, path);
243 return (1);
245 dbname = CSCOPE_DBFILE;
246 } else if ((dbname = strrchr(dname, '/')) != NULL)
247 *dbname++ = '\0';
249 /* Allocate a cscope connection structure and initialize its fields. */
250 len = strlen(dname);
251 CALLOC_RET(sp, csc, CSC *, 1, sizeof(CSC) + len);
252 csc->dname = csc->buf;
253 csc->dlen = len;
254 memcpy(csc->dname, dname, len);
255 csc->mtime = sb.st_mtime;
257 /* Get the search paths for the cscope. */
258 if (get_paths(sp, csc))
259 goto err;
261 /* Start the cscope process. */
262 if (run_cscope(sp, csc, dbname))
263 goto err;
266 * Add the cscope connection to the screen's list. From now on,
267 * on error, we have to call terminate, which expects the csc to
268 * be on the chain.
270 LIST_INSERT_HEAD(&exp->cscq, csc, q);
272 /* Read the initial prompt from the cscope to make sure it's okay. */
273 if (read_prompt(sp, csc)) {
274 terminate(sp, csc, 0);
275 return (1);
278 return (0);
280 err: free(csc);
281 return (1);
285 * get_paths --
286 * Get the directories to search for the files associated with this
287 * cscope database.
289 static int
290 get_paths(sp, csc)
291 SCR *sp;
292 CSC *csc;
294 struct stat sb;
295 int fd, nentries;
296 size_t len;
297 char *p, **pathp, buf[MAXPATHLEN * 2];
300 * EXTENSION #2:
302 * If there's a cscope directory with a file named CSCOPE_PATHS, it
303 * contains a colon-separated list of paths in which to search for
304 * files returned by cscope.
306 * XXX
307 * These paths are absolute paths, and not relative to the cscope
308 * directory. To fix this, rewrite the each path using the cscope
309 * directory as a prefix.
311 (void)snprintf(buf, sizeof(buf), "%s/%s", csc->dname, CSCOPE_PATHS);
312 if (stat(buf, &sb) == 0) {
313 /* Read in the CSCOPE_PATHS file. */
314 len = sb.st_size;
315 MALLOC_RET(sp, csc->pbuf, char *, len + 1);
316 if ((fd = open(buf, O_RDONLY, 0)) < 0 ||
317 read(fd, csc->pbuf, len) != len) {
318 msgq_str(sp, M_SYSERR, buf, "%s");
319 if (fd >= 0)
320 (void)close(fd);
321 return (1);
323 (void)close(fd);
324 csc->pbuf[len] = '\0';
326 /* Count up the entries. */
327 for (nentries = 0, p = csc->pbuf; *p != '\0'; ++p)
328 if (p[0] == ':' && p[1] != '\0')
329 ++nentries;
331 /* Build an array of pointers to the paths. */
332 CALLOC_GOTO(sp,
333 csc->paths, char **, nentries + 1, sizeof(char **));
334 for (pathp = csc->paths, p = strtok(csc->pbuf, ":");
335 p != NULL; p = strtok(NULL, ":"))
336 *pathp++ = p;
337 return (0);
341 * If the CSCOPE_PATHS file doesn't exist, we look for files
342 * relative to the cscope directory.
344 if ((csc->pbuf = strdup(csc->dname)) == NULL) {
345 msgq(sp, M_SYSERR, NULL);
346 return (1);
348 CALLOC_GOTO(sp, csc->paths, char **, 2, sizeof(char *));
349 csc->paths[0] = csc->pbuf;
350 return (0);
352 alloc_err:
353 if (csc->pbuf != NULL) {
354 free(csc->pbuf);
355 csc->pbuf = NULL;
357 return (1);
361 * run_cscope --
362 * Fork off the cscope process.
364 static int
365 run_cscope(sp, csc, dbname)
366 SCR *sp;
367 CSC *csc;
368 char *dbname;
370 int to_cs[2], from_cs[2];
371 char cmd[MAXPATHLEN * 2];
374 * Cscope reads from to_cs[0] and writes to from_cs[1]; vi reads from
375 * from_cs[0] and writes to to_cs[1].
377 to_cs[0] = to_cs[1] = from_cs[0] = from_cs[0] = -1;
378 if (pipe(to_cs) < 0 || pipe(from_cs) < 0) {
379 msgq(sp, M_SYSERR, "pipe");
380 goto err;
382 switch (csc->pid = vfork()) {
383 case -1:
384 msgq(sp, M_SYSERR, "vfork");
385 err: if (to_cs[0] != -1)
386 (void)close(to_cs[0]);
387 if (to_cs[1] != -1)
388 (void)close(to_cs[1]);
389 if (from_cs[0] != -1)
390 (void)close(from_cs[0]);
391 if (from_cs[1] != -1)
392 (void)close(from_cs[1]);
393 return (1);
394 case 0: /* child: run cscope. */
395 (void)dup2(to_cs[0], STDIN_FILENO);
396 (void)dup2(from_cs[1], STDOUT_FILENO);
397 (void)dup2(from_cs[1], STDERR_FILENO);
399 /* Close unused file descriptors. */
400 (void)close(to_cs[1]);
401 (void)close(from_cs[0]);
403 /* Run the cscope command. */
404 #define CSCOPE_CMD_FMT "cd '%s' && exec cscope -dl -f %s"
405 (void)snprintf(cmd, sizeof(cmd),
406 CSCOPE_CMD_FMT, csc->dname, dbname);
407 (void)execl(_PATH_BSHELL, "sh", "-c", cmd, NULL);
408 msgq_str(sp, M_SYSERR, cmd, "execl: %s");
409 _exit (127);
410 /* NOTREACHED */
411 default: /* parent. */
412 /* Close unused file descriptors. */
413 (void)close(to_cs[0]);
414 (void)close(from_cs[1]);
417 * Save the file descriptors for later duplication, and
418 * reopen as streams.
420 csc->to_fd = to_cs[1];
421 csc->to_fp = fdopen(to_cs[1], "w");
422 csc->from_fd = from_cs[0];
423 csc->from_fp = fdopen(from_cs[0], "r");
424 break;
426 return (0);
430 * cscope_find --
431 * The cscope find command.
433 static int
434 cscope_find(sp, cmdp, pattern)
435 SCR *sp;
436 EXCMD *cmdp;
437 char *pattern;
439 CSC *csc, *csc_next;
440 EX_PRIVATE *exp;
441 FREF *frp;
442 TAGQ *rtqp, *tqp;
443 TAG *rtp;
444 recno_t lno;
445 size_t cno, search;
446 int force, istmp, matches;
448 exp = EXP(sp);
450 /* Check for connections. */
451 if (exp->cscq.lh_first == NULL) {
452 msgq(sp, M_ERR, "310|No cscope connections running");
453 return (1);
457 * Allocate all necessary memory before doing anything hard. If the
458 * tags stack is empty, we'll need the `local context' TAGQ structure
459 * later.
461 rtp = NULL;
462 rtqp = NULL;
463 if (exp->tq.cqh_first == (void *)&exp->tq) {
464 /* Initialize the `local context' tag queue structure. */
465 CALLOC_GOTO(sp, rtqp, TAGQ *, 1, sizeof(TAGQ));
466 CIRCLEQ_INIT(&rtqp->tagq);
468 /* Initialize and link in its tag structure. */
469 CALLOC_GOTO(sp, rtp, TAG *, 1, sizeof(TAG));
470 CIRCLEQ_INSERT_HEAD(&rtqp->tagq, rtp, q);
471 rtqp->current = rtp;
474 /* Create the cscope command. */
475 if ((tqp = create_cs_cmd(sp, pattern, &search)) == NULL)
476 goto err;
479 * Stick the current context in a convenient place, we'll lose it
480 * when we switch files.
482 frp = sp->frp;
483 lno = sp->lno;
484 cno = sp->cno;
485 istmp = F_ISSET(sp->frp, FR_TMPFILE) && !F_ISSET(cmdp, E_NEWSCREEN);
487 /* Search all open connections for a match. */
488 matches = 0;
489 for (csc = exp->cscq.lh_first; csc != NULL; csc = csc_next) {
490 /* Copy csc->q.lh_next here in case csc is killed. */
491 csc_next = csc->q.le_next;
494 * Send the command to the cscope program. (We skip the
495 * first two bytes of the command, because we stored the
496 * search cscope command character and a leading space
497 * there.)
499 (void)fprintf(csc->to_fp, "%d%s\n", search, tqp->tag + 2);
500 (void)fflush(csc->to_fp);
502 /* Read the output. */
503 if (parse(sp, csc, tqp, &matches)) {
504 if (rtqp != NULL)
505 free(rtqp);
506 tagq_free(sp, tqp);
507 return (1);
511 if (matches == 0) {
512 msgq(sp, M_INFO, "278|No matches for query");
513 return (0);
516 tqp->current = tqp->tagq.cqh_first;
518 /* Try to switch to the first tag. */
519 force = FL_ISSET(cmdp->iflags, E_C_FORCE);
520 if (F_ISSET(cmdp, E_NEWSCREEN)) {
521 if (ex_tag_Nswitch(sp, tqp->current, force))
522 goto err;
524 /* Everything else gets done in the new screen. */
525 sp = sp->nextdisp;
526 exp = EXP(sp);
527 } else
528 if (ex_tag_nswitch(sp, tqp->current, force))
529 goto err;
532 * If this is the first tag, put a `current location' queue entry
533 * in place, so we can pop all the way back to the current mark.
534 * Note, it doesn't point to much of anything, it's a placeholder.
536 if (exp->tq.cqh_first == (void *)&exp->tq) {
537 CIRCLEQ_INSERT_HEAD(&exp->tq, rtqp, q);
538 } else
539 rtqp = exp->tq.cqh_first;
541 /* Link the current TAGQ structure into place. */
542 CIRCLEQ_INSERT_HEAD(&exp->tq, tqp, q);
544 (void)cscope_search(sp, tqp, tqp->current);
547 * Move the current context from the temporary save area into the
548 * right structure.
550 * If we were in a temporary file, we don't have a context to which
551 * we can return, so just make it be the same as what we're moving
552 * to. It will be a little odd that ^T doesn't change anything, but
553 * I don't think it's a big deal.
555 if (istmp) {
556 rtqp->current->frp = sp->frp;
557 rtqp->current->lno = sp->lno;
558 rtqp->current->cno = sp->cno;
559 } else {
560 rtqp->current->frp = frp;
561 rtqp->current->lno = lno;
562 rtqp->current->cno = cno;
565 return (0);
567 err:
568 alloc_err:
569 if (rtqp != NULL)
570 free(rtqp);
571 if (rtp != NULL)
572 free(rtp);
573 return (1);
577 * create_cs_cmd --
578 * Build a cscope command, creating and initializing the base TAGQ.
580 static TAGQ *
581 create_cs_cmd(sp, pattern, searchp)
582 SCR *sp;
583 char *pattern;
584 size_t *searchp;
586 CB *cbp;
587 TAGQ *tqp;
588 size_t tlen;
589 char *p;
592 * Cscope supports a "change pattern" command which we never use,
593 * cscope command 5. Set CSCOPE_QUERIES[5] to " " since the user
594 * can't pass " " as the first character of pattern. That way the
595 * user can't ask for pattern 5 so we don't need any special-case
596 * code.
598 #define CSCOPE_QUERIES "sgdct efi"
600 if (pattern == NULL)
601 goto usage;
603 /* Skip leading blanks, check for command character. */
604 for (; isblank(pattern[0]); ++pattern);
605 if (pattern[0] == '\0' || !isblank(pattern[1]))
606 goto usage;
607 for (*searchp = 0, p = CSCOPE_QUERIES;
608 *p != '\0' && *p != pattern[0]; ++*searchp, ++p);
609 if (*p == '\0') {
610 msgq(sp, M_ERR,
611 "311|%s: unknown search type: use one of %s",
612 KEY_NAME(sp, pattern[0]), CSCOPE_QUERIES);
613 return (NULL);
616 /* Skip <blank> characters to the pattern. */
617 for (p = pattern + 1; *p != '\0' && isblank(*p); ++p);
618 if (*p == '\0') {
619 usage: (void)csc_help(sp, "find");
620 return (NULL);
623 /* The user can specify the contents of a buffer as the pattern. */
624 cbp = NULL;
625 if (p[0] == '"' && p[1] != '\0' && p[2] == '\0')
626 CBNAME(sp, cbp, p[1]);
627 if (cbp != NULL) {
628 p = cbp->textq.cqh_first->lb;
629 tlen = cbp->textq.cqh_first->len;
630 } else
631 tlen = strlen(p);
633 /* Allocate and initialize the TAGQ structure. */
634 CALLOC(sp, tqp, TAGQ *, 1, sizeof(TAGQ) + tlen + 3);
635 if (tqp == NULL)
636 return (NULL);
637 CIRCLEQ_INIT(&tqp->tagq);
638 tqp->tag = tqp->buf;
639 tqp->tag[0] = pattern[0];
640 tqp->tag[1] = ' ';
641 tqp->tlen = tlen + 2;
642 memcpy(tqp->tag + 2, p, tlen);
643 tqp->tag[tlen + 2] = '\0';
644 F_SET(tqp, TAG_CSCOPE);
646 return (tqp);
650 * parse --
651 * Parse the cscope output.
653 static int
654 parse(sp, csc, tqp, matchesp)
655 SCR *sp;
656 CSC *csc;
657 TAGQ *tqp;
658 int *matchesp;
660 TAG *tp;
661 recno_t slno;
662 size_t dlen, nlen, slen;
663 int ch, i, isolder, nlines;
664 char *dname, *name, *search, *p, *t, dummy[2], buf[2048];
666 for (;;) {
667 if (!fgets(buf, sizeof(buf), csc->from_fp))
668 goto io_err;
671 * If the database is out of date, or there's some other
672 * problem, cscope will output error messages before the
673 * number-of-lines output. Display/discard any output
674 * that doesn't match what we want.
676 #define CSCOPE_NLINES_FMT "cscope: %d lines%1[\n]"
677 if (sscanf(buf, CSCOPE_NLINES_FMT, &nlines, dummy) == 2)
678 break;
679 if ((p = strchr(buf, '\n')) != NULL)
680 *p = '\0';
681 msgq(sp, M_ERR, "%s: \"%s\"", csc->dname, buf);
684 while (nlines--) {
685 if (fgets(buf, sizeof(buf), csc->from_fp) == NULL)
686 goto io_err;
688 /* If the line's too long for the buffer, discard it. */
689 if ((p = strchr(buf, '\n')) == NULL) {
690 while ((ch = getc(csc->from_fp)) != EOF && ch != '\n');
691 continue;
693 *p = '\0';
696 * The cscope output is in the following format:
698 * <filename> <context> <line number> <pattern>
700 * Figure out how long everything is so we can allocate in one
701 * swell foop, but discard anything that looks wrong.
703 for (p = buf, i = 0;
704 i < 3 && (t = strsep(&p, "\t ")) != NULL; ++i)
705 switch (i) {
706 case 0: /* Filename. */
707 name = t;
708 nlen = strlen(name);
709 break;
710 case 1: /* Context. */
711 break;
712 case 2: /* Line number. */
713 slno = (recno_t)atol(t);
714 break;
716 if (i != 3 || p == NULL || t == NULL)
717 continue;
719 /* The rest of the string is the search pattern. */
720 search = p;
721 slen = strlen(p);
723 /* Resolve the file name. */
724 csc_file(sp, csc, name, &dname, &dlen, &isolder);
727 * If the file is older than the cscope database, that is,
728 * the database was built since the file was last modified,
729 * or there wasn't a search string, use the line number.
731 if (isolder || strcmp(search, "<unknown>") == 0) {
732 search = NULL;
733 slen = 0;
737 * Allocate and initialize a tag structure plus the variable
738 * length cscope information that follows it.
740 CALLOC_RET(sp, tp,
741 TAG *, 1, sizeof(TAG) + dlen + 2 + nlen + 1 + slen + 1);
742 tp->fname = tp->buf;
743 if (dlen != 0) {
744 memcpy(tp->fname, dname, dlen);
745 tp->fname[dlen] = '/';
746 ++dlen;
748 memcpy(tp->fname + dlen, name, nlen + 1);
749 tp->fnlen = dlen + nlen;
750 tp->slno = slno;
751 if (slen != 0) {
752 tp->search = tp->fname + tp->fnlen + 1;
753 memcpy(tp->search, search, (tp->slen = slen) + 1);
755 CIRCLEQ_INSERT_TAIL(&tqp->tagq, tp, q);
757 ++*matchesp;
760 (void)read_prompt(sp, csc);
761 return (0);
763 io_err: if (feof(csc->from_fp))
764 errno = EIO;
765 msgq_str(sp, M_SYSERR, "%s", csc->dname);
766 terminate(sp, csc, 0);
767 return (1);
771 * csc_file --
772 * Search for the right path to this file.
774 static void
775 csc_file(sp, csc, name, dirp, dlenp, isolderp)
776 SCR *sp;
777 CSC *csc;
778 char *name, **dirp;
779 size_t *dlenp;
780 int *isolderp;
782 struct stat sb;
783 char **pp, buf[MAXPATHLEN];
786 * Check for the file in all of the listed paths. If we don't
787 * find it, we simply return it unchanged. We have to do this
788 * now, even though it's expensive, because if the user changes
789 * directories, we can't change our minds as to where the file
790 * lives.
792 for (pp = csc->paths; *pp != NULL; ++pp) {
793 (void)snprintf(buf, sizeof(buf), "%s/%s", *pp, name);
794 if (stat(buf, &sb) == 0) {
795 *dirp = *pp;
796 *dlenp = strlen(*pp);
797 *isolderp = sb.st_mtime < csc->mtime;
798 return;
801 *dlenp = 0;
805 * cscope_help --
806 * The cscope help command.
808 static int
809 cscope_help(sp, cmdp, subcmd)
810 SCR *sp;
811 EXCMD *cmdp;
812 char *subcmd;
814 return (csc_help(sp, subcmd));
818 * csc_help --
819 * Display help/usage messages.
821 static int
822 csc_help(sp, cmd)
823 SCR *sp;
824 char *cmd;
826 CC const *ccp;
828 if (cmd != NULL && *cmd != '\0')
829 if ((ccp = lookup_ccmd(cmd)) == NULL) {
830 ex_printf(sp,
831 "%s doesn't match any cscope command\n", cmd);
832 return (1);
833 } else {
834 ex_printf(sp,
835 "Command: %s (%s)\n", ccp->name, ccp->help_msg);
836 ex_printf(sp, " Usage: %s\n", ccp->usage_msg);
837 return (0);
840 ex_printf(sp, "cscope commands:\n");
841 for (ccp = cscope_cmds; ccp->name != NULL; ++ccp)
842 ex_printf(sp, " %*s: %s\n", 5, ccp->name, ccp->help_msg);
843 return (0);
847 * cscope_kill --
848 * The cscope kill command.
850 static int
851 cscope_kill(sp, cmdp, cn)
852 SCR *sp;
853 EXCMD *cmdp;
854 char *cn;
856 return (terminate(sp, NULL, atoi(cn)));
860 * terminate --
861 * Detach from a cscope process.
863 static int
864 terminate(sp, csc, n)
865 SCR *sp;
866 CSC *csc;
867 int n;
869 EX_PRIVATE *exp;
870 int i, pstat;
872 exp = EXP(sp);
875 * We either get a csc structure or a number. If not provided a
876 * csc structure, find the right one.
878 if (csc == NULL) {
879 if (n < 1)
880 goto badno;
881 for (i = 1, csc = exp->cscq.lh_first;
882 csc != NULL; csc = csc->q.le_next, i++)
883 if (i == n)
884 break;
885 if (csc == NULL) {
886 badno: msgq(sp, M_ERR, "312|%d: no such cscope session", n);
887 return (1);
892 * XXX
893 * Theoretically, we have the only file descriptors to the process,
894 * so closing them should let it exit gracefully, deleting temporary
895 * files, etc. The original vi cscope integration sent the cscope
896 * connection a SIGTERM signal, so I'm not sure if closing the file
897 * descriptors is sufficient.
899 if (csc->from_fp != NULL)
900 (void)fclose(csc->from_fp);
901 if (csc->to_fp != NULL)
902 (void)fclose(csc->to_fp);
903 (void)waitpid(csc->pid, &pstat, 0);
905 /* Discard cscope connection information. */
906 LIST_REMOVE(csc, q);
907 if (csc->pbuf != NULL)
908 free(csc->pbuf);
909 if (csc->paths != NULL)
910 free(csc->paths);
911 free(csc);
912 return (0);
916 * cscope_reset --
917 * The cscope reset command.
919 static int
920 cscope_reset(sp, cmdp, notusedp)
921 SCR *sp;
922 EXCMD *cmdp;
923 char *notusedp;
925 EX_PRIVATE *exp;
927 for (exp = EXP(sp); exp->cscq.lh_first != NULL;)
928 if (cscope_kill(sp, cmdp, "1"))
929 return (1);
930 return (0);
934 * cscope_display --
935 * Display current connections.
937 * PUBLIC: int cscope_display __P((SCR *));
940 cscope_display(sp)
941 SCR *sp;
943 EX_PRIVATE *exp;
944 CSC *csc;
945 int i;
947 exp = EXP(sp);
948 if (exp->cscq.lh_first == NULL) {
949 ex_printf(sp, "No cscope connections.\n");
950 return (0);
952 for (i = 1,
953 csc = exp->cscq.lh_first; csc != NULL; ++i, csc = csc->q.le_next)
954 ex_printf(sp,
955 "%2d %s (process %lu)\n", i, csc->dname, (u_long)csc->pid);
956 return (0);
960 * cscope_search --
961 * Search a file for a cscope entry.
963 * PUBLIC: int cscope_search __P((SCR *, TAGQ *, TAG *));
966 cscope_search(sp, tqp, tp)
967 SCR *sp;
968 TAGQ *tqp;
969 TAG *tp;
971 MARK m;
973 /* If we don't have a search pattern, use the line number. */
974 if (tp->search == NULL) {
975 if (!db_exist(sp, tp->slno)) {
976 tag_msg(sp, TAG_BADLNO, tqp->tag);
977 return (1);
979 m.lno = tp->slno;
980 } else {
982 * Search for the tag; cheap fallback for C functions
983 * if the name is the same but the arguments have changed.
985 m.lno = 1;
986 m.cno = 0;
987 if (f_search(sp, &m, &m,
988 tp->search, tp->slen, NULL, SEARCH_CSCOPE | SEARCH_FILE)) {
989 tag_msg(sp, TAG_SEARCH, tqp->tag);
990 return (1);
994 * !!!
995 * Historically, tags set the search direction if it wasn't
996 * already set.
998 if (sp->searchdir == NOTSET)
999 sp->searchdir = FORWARD;
1003 * !!!
1004 * Tags move to the first non-blank, NOT the search pattern start.
1006 sp->lno = m.lno;
1007 sp->cno = 0;
1008 (void)nonblank(sp, sp->lno, &sp->cno);
1009 return (0);
1014 * lookup_ccmd --
1015 * Return a pointer to the command structure.
1017 static CC const *
1018 lookup_ccmd(name)
1019 char *name;
1021 CC const *ccp;
1022 size_t len;
1024 len = strlen(name);
1025 for (ccp = cscope_cmds; ccp->name != NULL; ++ccp)
1026 if (strncmp(name, ccp->name, len) == 0)
1027 return (ccp);
1028 return (NULL);
1032 * read_prompt --
1033 * Read a prompt from cscope.
1035 static int
1036 read_prompt(sp, csc)
1037 SCR *sp;
1038 CSC *csc;
1040 int ch;
1042 #define CSCOPE_PROMPT ">> "
1043 for (;;) {
1044 while ((ch =
1045 getc(csc->from_fp)) != EOF && ch != CSCOPE_PROMPT[0]);
1046 if (ch == EOF) {
1047 terminate(sp, csc, 0);
1048 return (1);
1050 if (getc(csc->from_fp) != CSCOPE_PROMPT[1])
1051 continue;
1052 if (getc(csc->from_fp) != CSCOPE_PROMPT[2])
1053 continue;
1054 break;
1056 return (0);