summaryrefslogtreecommitdiff
path: root/Userland/Utilities/man.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Userland/Utilities/man.cpp')
-rw-r--r--Userland/Utilities/man.cpp57
1 files changed, 55 insertions, 2 deletions
diff --git a/Userland/Utilities/man.cpp b/Userland/Utilities/man.cpp
index 28b637649e..925dfe27e8 100644
--- a/Userland/Utilities/man.cpp
+++ b/Userland/Utilities/man.cpp
@@ -10,10 +10,39 @@
#include <LibCore/ArgsParser.h>
#include <LibCore/File.h>
#include <LibMarkdown/Document.h>
+#include <fcntl.h>
+#include <spawn.h>
#include <stdio.h>
#include <sys/ioctl.h>
+#include <sys/wait.h>
#include <unistd.h>
+static pid_t pipe_to_pager(String const& command)
+{
+ char const* argv[] = { "sh", "-c", command.characters(), nullptr };
+
+ int stdout_pipe[2] = {};
+ if (pipe2(stdout_pipe, O_CLOEXEC)) {
+ perror("pipe2");
+ exit(1);
+ }
+ posix_spawn_file_actions_t action;
+ posix_spawn_file_actions_init(&action);
+ posix_spawn_file_actions_adddup2(&action, stdout_pipe[0], STDIN_FILENO);
+
+ pid_t pid;
+ if ((errno = posix_spawnp(&pid, argv[0], &action, nullptr, const_cast<char**>(argv), environ))) {
+ perror("posix_spawn");
+ exit(1);
+ }
+ posix_spawn_file_actions_destroy(&action);
+
+ dup2(stdout_pipe[1], STDOUT_FILENO);
+ close(stdout_pipe[1]);
+ close(stdout_pipe[0]);
+ return pid;
+}
+
int main(int argc, char* argv[])
{
int view_width = 0;
@@ -26,7 +55,7 @@ int main(int argc, char* argv[])
if (view_width == 0)
view_width = 80;
- if (pledge("stdio rpath", nullptr) < 0) {
+ if (pledge("stdio rpath exec proc", nullptr) < 0) {
perror("pledge");
return 1;
}
@@ -36,15 +65,22 @@ int main(int argc, char* argv[])
return 1;
}
+ if (unveil("/bin", "x") < 0) {
+ perror("unveil");
+ return 1;
+ }
+
unveil(nullptr, nullptr);
const char* section = nullptr;
const char* name = nullptr;
+ const char* pager = nullptr;
Core::ArgsParser args_parser;
args_parser.set_general_help("Read manual pages. Try 'man man' to get started.");
args_parser.add_positional_argument(section, "Section of the man page", "section", Core::ArgsParser::Required::No);
args_parser.add_positional_argument(name, "Name of the man page", "name");
+ args_parser.add_option(pager, "Pager to pipe the man page to", "pager", 'P', "pager");
args_parser.parse(argc, argv);
@@ -78,12 +114,23 @@ int main(int argc, char* argv[])
auto file = Core::File::construct();
file->set_filename(make_path(section));
+ String pager_command = pager;
+ if (!pager) {
+ String clean_name(name);
+ String clean_section(section);
+
+ clean_name.replace("'", "'\\''");
+ clean_section.replace("'", "'\\''");
+ pager_command = String::formatted("less -P 'Manual Page {}({}) line %l?e (END):.'", clean_name, clean_section);
+ }
+ pid_t pager_pid = pipe_to_pager(pager_command);
+
if (!file->open(Core::OpenMode::ReadOnly)) {
perror("Failed to open man page file");
exit(1);
}
- if (pledge("stdio", nullptr) < 0) {
+ if (pledge("stdio proc", nullptr) < 0) {
perror("pledge");
return 1;
}
@@ -99,4 +146,10 @@ int main(int argc, char* argv[])
String rendered = document->render_for_terminal(view_width);
out("{}", rendered);
+
+ // FIXME: Remove this wait, it shouldn't be necessary but Shell does not
+ // resume properly without it. This wait also breaks <C-z> backgrounding
+ fclose(stdout);
+ int wstatus;
+ waitpid(pager_pid, &wstatus, 0);
}