2008-06-22 02:12:16 +02:00
|
|
|
--- src/man.c
|
2008-11-28 15:35:17 +01:00
|
|
|
+++ src/man.c 2008-11-28 13:12:22.171541000 +0100
|
2010-04-30 13:51:20 +02:00
|
|
|
@@ -3223,12 +3223,132 @@ static int locate_page (const char *manp
|
2008-06-22 02:12:16 +02:00
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
+#ifndef PROMPT_IF_MULTIPLE_SECTIONS
|
|
|
|
+#define PROMPT_IF_MULTIPLE_SECTIONS 2 /* 0: No prompt; 1: Show possible sections; 2: Do prompt for */
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 1)
|
|
|
|
+static sig_atomic_t expired;
|
|
|
|
+static void handler(int sig)
|
|
|
|
+{
|
|
|
|
+ (void)sig;
|
|
|
|
+ expired++;
|
|
|
|
+}
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
static int display_pages (struct candidate *candidates)
|
|
|
|
{
|
|
|
|
struct candidate *candp;
|
|
|
|
int found = 0;
|
2008-11-28 15:35:17 +01:00
|
|
|
+ int plain = 0;
|
|
|
|
+
|
2008-06-22 02:12:16 +02:00
|
|
|
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 1)
|
|
|
|
+ char reqsect[64] = { 0 };
|
2008-11-28 15:35:17 +01:00
|
|
|
+ ssize_t len = 0;
|
2008-06-22 02:12:16 +02:00
|
|
|
+ do {
|
2008-11-28 15:35:17 +01:00
|
|
|
+ const char *lext;
|
2008-06-22 02:12:16 +02:00
|
|
|
+ struct sigaction sa;
|
|
|
|
+ int used = 0x2A;
|
|
|
|
+
|
|
|
|
+ if (findall)
|
|
|
|
+ break;
|
|
|
|
+ if (external)
|
|
|
|
+ break;
|
|
|
|
+ if ((troff + catman + (print_where || print_where_cat)))
|
|
|
|
+ break;
|
|
|
|
+ if (getenv("MAN_POSIXLY_CORRECT"))
|
|
|
|
+ break;
|
|
|
|
+ if (!isatty(STDOUT_FILENO))
|
|
|
|
+ break;
|
|
|
|
+ if (!isatty(STDERR_FILENO))
|
|
|
|
+ break;
|
|
|
|
+ if (!isatty(STDIN_FILENO))
|
|
|
|
+ break;
|
|
|
|
+ if (candidates->next == (struct candidate*)0)
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ fputs("Man: ", stderr);
|
|
|
|
+ fputs(_("find all matching manual pages"), stderr);
|
2010-04-30 13:51:20 +02:00
|
|
|
+ fputs(" (set MAN_POSIXLY_CORRECT to avoid this)", stderr);
|
2008-06-22 02:12:16 +02:00
|
|
|
+ fputc('\n', stderr);
|
|
|
|
+
|
2008-11-28 15:35:17 +01:00
|
|
|
+ lext = NULL;
|
2008-06-22 02:12:16 +02:00
|
|
|
+ for (candp = candidates; candp; candp = candp->next) {
|
2008-11-28 15:35:17 +01:00
|
|
|
+ const struct mandata *info = candp->source;
|
|
|
|
+ const char *base = "";
|
|
|
|
+
|
|
|
|
+ if (lext && STREQ(lext, info->ext))
|
|
|
|
+ base = "+";
|
|
|
|
+
|
|
|
|
+ fprintf(stderr, " %c %s (%s%s)", used, info->name, info->ext, base);
|
2008-06-22 02:12:16 +02:00
|
|
|
+ if (info->whatis)
|
|
|
|
+ fprintf(stderr, "\t%s", info->whatis);
|
|
|
|
+ fputc('\n', stderr);
|
2008-11-28 15:35:17 +01:00
|
|
|
+ lext = info->ext;
|
2008-06-22 02:12:16 +02:00
|
|
|
+ used = ' ';
|
|
|
|
+ }
|
|
|
|
+ fputs("Man: ", stderr);
|
|
|
|
+ fputs(_("What manual page do you want?\n"), stderr);
|
|
|
|
+ fputs("Man: ", stderr);
|
|
|
|
+ fflush(stderr);
|
|
|
|
+
|
|
|
|
+ sigemptyset(&sa.sa_mask);
|
|
|
|
+ sa.sa_flags = SA_RESETHAND;
|
|
|
|
+ sa.sa_handler = handler;
|
|
|
|
+
|
|
|
|
+ sigaction(SIGALRM, &sa, (struct sigaction*)0);
|
|
|
|
+
|
|
|
|
+ alarm(7);
|
|
|
|
+ len = read(STDIN_FILENO, reqsect, sizeof(reqsect)-1);
|
|
|
|
+ alarm(0);
|
|
|
|
+
|
|
|
|
+ if (expired) {
|
|
|
|
+ tcflush(STDIN_FILENO, TCIFLUSH);
|
|
|
|
+ fputc('\n', stderr);
|
|
|
|
+ fflush(stderr);
|
|
|
|
+ len = 0;
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ if (len > 0) {
|
|
|
|
+ char * end;
|
|
|
|
+ if ((end = strchr(reqsect, '\n')))
|
|
|
|
+ *end = '\0';
|
2008-11-28 15:35:17 +01:00
|
|
|
+ if ((end = strchr(reqsect, '+'))) {
|
|
|
|
+ plain = 1;
|
|
|
|
+ *end = '\0';
|
|
|
|
+ }
|
2008-06-22 02:12:16 +02:00
|
|
|
+ reqsect[len] = '\0';
|
|
|
|
+ len = (ssize_t)strlen(reqsect);
|
2008-11-28 15:35:17 +01:00
|
|
|
+
|
2008-06-22 02:12:16 +02:00
|
|
|
+ for (candp = candidates; candp; candp = candp->next) {
|
|
|
|
+ if (len == 0)
|
|
|
|
+ break;
|
2008-11-28 15:35:17 +01:00
|
|
|
+ if (plain) {
|
|
|
|
+ const char *base = strrchr(candp->path, '/');
|
|
|
|
+ if (base && !STREQ(base, "/man"))
|
|
|
|
+ continue;
|
|
|
|
+ }
|
2008-06-22 02:12:16 +02:00
|
|
|
+ if ((found = STREQ(reqsect, candp->source->ext)))
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ if (!found)
|
|
|
|
+ len = 0;
|
|
|
|
+ found = 0;
|
|
|
|
+ }
|
|
|
|
+ } while (0);
|
|
|
|
+#endif
|
2008-11-28 15:35:17 +01:00
|
|
|
|
2008-06-22 02:12:16 +02:00
|
|
|
for (candp = candidates; candp; candp = candp->next) {
|
|
|
|
+
|
|
|
|
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 1)
|
2008-11-28 15:35:17 +01:00
|
|
|
+ if (plain) {
|
|
|
|
+ const char *base = strrchr(candp->path, '/');
|
|
|
|
+ if (base && !STREQ(base, "/man"))
|
|
|
|
+ continue;
|
|
|
|
+ }
|
2008-06-22 02:12:16 +02:00
|
|
|
+ if (len && !STREQ(reqsect, candp->source->ext))
|
|
|
|
+ continue;
|
|
|
|
+#endif
|
|
|
|
global_manpath = is_global_mandir (candp->path);
|
|
|
|
if (!global_manpath)
|
|
|
|
drop_effective_privs ();
|
2010-04-30 13:51:20 +02:00
|
|
|
@@ -3250,9 +3370,51 @@ static int display_pages (struct candida
|
2008-06-22 02:12:16 +02:00
|
|
|
regain_effective_privs ();
|
|
|
|
|
|
|
|
if (found && !findall)
|
|
|
|
- return found;
|
|
|
|
+ {
|
|
|
|
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 0)
|
|
|
|
+ if (external)
|
|
|
|
+ goto out;
|
|
|
|
+ if ((troff + catman + (print_where || print_where_cat)))
|
|
|
|
+ goto out;
|
|
|
|
+# if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS == 1)
|
|
|
|
+ if (getenv("MAN_POSIXLY_CORRECT"))
|
|
|
|
+ goto out;
|
|
|
|
+# endif
|
|
|
|
+ if (!isatty(STDOUT_FILENO))
|
|
|
|
+ goto out;
|
|
|
|
+ if (!isatty(STDERR_FILENO))
|
|
|
|
+ goto out;
|
|
|
|
+ /*
|
|
|
|
+ * Should be able to use the output as done by whatis(1)
|
|
|
|
+ */
|
|
|
|
+# if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS == 1)
|
|
|
|
+ if (candp->next)
|
|
|
|
+# else
|
|
|
|
+ if (candp->next && getenv("MAN_POSIXLY_CORRECT"))
|
|
|
|
+# endif
|
|
|
|
+ {
|
|
|
|
+ int used = 0x2A;
|
|
|
|
+
|
|
|
|
+ fputs("Man: ", stderr);
|
|
|
|
+ fputs(_("find all matching manual pages"), stderr);
|
|
|
|
+ fputc('\n', stderr);
|
|
|
|
+
|
|
|
|
+ do {
|
|
|
|
+ struct mandata *info = candp->source;
|
|
|
|
+ fprintf(stderr, " %c %s (%s)", used, info->name, info->ext);
|
|
|
|
+ if (info->whatis) {
|
|
|
|
+ fprintf(stderr, "\t%s", info->whatis);
|
|
|
|
+ }
|
|
|
|
+ fputc('\n', stderr);
|
|
|
|
+ used = ' ';
|
|
|
|
+ } while ((candp = candp->next));
|
|
|
|
+ fflush(stderr);
|
|
|
|
+ }
|
|
|
|
+#endif
|
|
|
|
+ goto out;
|
|
|
|
+ }
|
|
|
|
}
|
|
|
|
-
|
|
|
|
+out:
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|