man/man-db-2.6.3-listall.dif
2024-11-11 09:11:28 +00:00

226 lines
5.7 KiB
Plaintext

---
man/man1/man.man1 | 8 ++
src/man.c | 180 +++++++++++++++++++++++++++++++++++++++++++++++++++++-
2 files changed, 186 insertions(+), 2 deletions(-)
--- man/man1/man.man1
+++ man/man1/man.man1 2022-08-17 13:11:29.974677243 +0000
@@ -1236,6 +1236,14 @@ However, some users want to see them any
.RB $ MAN_KEEP_STDERR
is set to any non-empty value, error output will be displayed as usual.
.TP
+.if !'po4a'hide' .BR MAN_POSIXLY_CORRECT
+If many man pages are available corresponding to the requested one,
+.B %man%
+will display them in a list, unless
+.RB $ MAN_POSIXLY_CORRECT
+is set, in which case the first page in the list will be displayed
+automatically.
+.TP
.if !'po4a'hide' .B MAN_DISABLE_SECCOMP
On Linux,
.B %man%
--- src/man.c
+++ src/man.c 2022-08-17 13:10:07.436205495 +0000
@@ -3651,12 +3651,141 @@ static int locate_page (const char *manp
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;
+ int plain = 0;
+
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 1)
+ char reqsect[64] = { 0 };
+ ssize_t len = 0;
+ int index = -1;
+ const char *lext;
+ do {
+ 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);
+ fputs(" (set MAN_POSIXLY_CORRECT to avoid this)", stderr);
+ fputc('\n', stderr);
+
+ lext = NULL;
+ for (candp = candidates; candp; candp = candp->next) {
+ const struct mandata *info = candp->source;
+
+ if (lext && STREQ(lext, info->ext))
+ fprintf(stderr, " %c %s (%s+%d)", used, info->name, info->ext, candp->add_index);
+ else
+ fprintf(stderr, " %c %s (%s)", used, info->name, info->ext);
+
+ if (info->whatis)
+ fprintf(stderr, "\t%s", info->whatis);
+ fputc('\n', stderr);
+ lext = info->ext;
+ 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';
+ if ((end = strchr(reqsect, '+'))) {
+ plain = 1;
+ *end = '\0';
+ if (*++end)
+ index = atoi(end);
+ }
+ reqsect[len] = '\0';
+ len = (ssize_t)strlen(reqsect);
+
+ for (candp = candidates; candp; candp = candp->next) {
+ if (len == 0)
+ break;
+ if (plain) {
+ const char *base = strrchr(candp->path, '/');
+ if (base && !STREQ(base, "/man"))
+ continue;
+ }
+ if (index >= 0 && index != candp->add_index)
+ continue;
+ if ((found = STREQ(reqsect, candp->source->ext)))
+ break;
+ }
+ if (!found)
+ len = 0;
+ found = 0;
+ }
+ } while (0);
+#endif
for (candp = candidates; candp; candp = candp->next) {
+
+#if defined(PROMPT_IF_MULTIPLE_SECTIONS) && (PROMPT_IF_MULTIPLE_SECTIONS > 1)
+ if (plain) {
+ const char *base = strrchr(candp->path, '/');
+ if (base && !STREQ(base, "/man"))
+ continue;
+ }
+ if (len) {
+ if (!STREQ(reqsect, candp->source->ext))
+ continue;
+ if (index >= 0 && index != candp->add_index)
+ continue;
+ }
+#endif
global_manpath = is_global_mandir (candp->path);
if (!global_manpath)
drop_effective_privs ();
@@ -3679,9 +3808,56 @@ static int display_pages (struct candida
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);
+
+ lext = NULL;
+ do {
+ struct mandata *info = candp->source;
+ if (lext && STREQ(lext, info->ext))
+ fprintf(stderr, " %c %s (%s+%d)", used, info->name, info->ext, candp->add_index);
+ else
+ fprintf(stderr, " %c %s (%s)", used, info->name, info->ext);
+ if (info->whatis) {
+ fprintf(stderr, "\t%s", info->whatis);
+ }
+ fputc('\n', stderr);
+ lext = info->ext;
+ used = ' ';
+ } while ((candp = candp->next));
+ fflush(stderr);
+ }
+#endif
+ goto out;
+ }
}
-
+out:
return found;
}