diff options
Diffstat (limited to 'src/libmain')
-rw-r--r-- | src/libmain/shared.cc | 40 | ||||
-rw-r--r-- | src/libmain/shared.hh | 12 |
2 files changed, 52 insertions, 0 deletions
diff --git a/src/libmain/shared.cc b/src/libmain/shared.cc index 9ac9d2773852..ba75847fdb63 100644 --- a/src/libmain/shared.cc +++ b/src/libmain/shared.cc @@ -285,4 +285,44 @@ int handleExceptions(const string & programName, std::function<void()> fun) } +RunPager::RunPager() +{ + string pager = getEnv("PAGER"); + if (!isatty(STDOUT_FILENO) || pager.empty()) return; + + /* Ignore SIGINT. The pager will handle it (and we'll get + SIGPIPE). */ + struct sigaction act; + act.sa_handler = SIG_IGN; + act.sa_flags = 0; + sigemptyset(&act.sa_mask); + if (sigaction(SIGINT, &act, 0)) throw SysError("ignoring SIGINT"); + + restoreSIGPIPE(); + + Pipe toPager; + toPager.create(); + + pid = startProcess([&]() { + if (dup2(toPager.readSide, STDIN_FILENO) == -1) + throw SysError("dupping stdin"); + execl("/bin/sh", "sh", "-c", pager.c_str(), NULL); + throw SysError(format("executing `%1%'") % pager); + }); + + if (dup2(toPager.writeSide, STDOUT_FILENO) == -1) + throw SysError("dupping stdout"); + +} + + +RunPager::~RunPager() +{ + if (pid != -1) { + close(STDOUT_FILENO); + pid.wait(true); + } +} + + } diff --git a/src/libmain/shared.hh b/src/libmain/shared.hh index c74e7cbc197d..c56203daefa2 100644 --- a/src/libmain/shared.hh +++ b/src/libmain/shared.hh @@ -69,6 +69,18 @@ template<class N> N getIntArg(const string & opt, /* Show the manual page for the specified program. */ void showManPage(const string & name); +/* The constructor of this class starts a pager if stdout is a + terminal and $PAGER is set. Stdout is redirected to the pager. */ +class RunPager +{ +public: + RunPager(); + ~RunPager(); + +private: + Pid pid; +}; + extern volatile ::sig_atomic_t blockInt; } |