summaryrefslogtreecommitdiff
path: root/main.cc
diff options
context:
space:
mode:
Diffstat (limited to 'main.cc')
-rw-r--r--main.cc74
1 files changed, 70 insertions, 4 deletions
diff --git a/main.cc b/main.cc
index 1ead6fd9..908b05b4 100644
--- a/main.cc
+++ b/main.cc
@@ -15,10 +15,18 @@ using namespace ledger;
#include <exception>
#include <iterator>
#include <string>
+#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <ctime>
+#ifdef HAVE_UNIX_PIPES
+#include <sys/types.h>
+#include <sys/wait.h>
+#include <unistd.h>
+#include "fdstream.hpp"
+#endif
+
#if !defined(DEBUG_LEVEL) || DEBUG_LEVEL <= RELEASE
#define auto_ptr bogus_auto_ptr
@@ -306,9 +314,57 @@ int parse_and_report(int argc, char * argv[], char * envp[])
// Configure the output stream
+#ifdef HAVE_UNIX_PIPES
+ int status, pfd[2]; // Pipe file descriptors
+#endif
std::ostream * out = &std::cout;
- if (! config.output_file.empty())
+
+ if (! config.output_file.empty()) {
out = new std::ofstream(config.output_file.c_str());
+ }
+#ifdef HAVE_UNIX_PIPES
+ else if (! config.pager.empty()) {
+ status = pipe(pfd);
+ if (status == -1)
+ throw error("Failed to create pipe");
+
+ status = fork();
+ if (status < 0) {
+ throw error("Failed to fork child process");
+ }
+ else if (status == 0) { // child
+ const char *arg0;
+
+ // Duplicate pipe's reading end into stdin
+ status = dup2(pfd[0], STDIN_FILENO);
+ if (status==-1)
+ perror("dup2");
+
+ // Close unuseful file descriptors: the pipe's writing and
+ // reading ends (the latter is not needed anymore, after the
+ // duplication).
+ close(pfd[1]);
+ close(pfd[0]);
+
+ // Find command name: its the substring starting right of the
+ // rightmost '/' character in the pager pathname. See manpage
+ // for strrchr.
+ arg0 = std::strrchr(config.pager.c_str(), '/');
+ if (arg0 != NULL)
+ arg0++;
+ else
+ arg0 = config.pager.c_str(); // No slashes in pager.
+
+ execlp(config.pager.c_str(), arg0, (char *)0);
+ perror("execl");
+ exit(1);
+ }
+ else { // parent
+ close(pfd[0]);
+ out = new boost::fdostream(pfd[1]);
+ }
+ }
+#endif
// Compile the format strings
@@ -446,7 +502,6 @@ def vmax(d, val):\n\
}
#if DEBUG_LEVEL >= BETA
-
clear_all_xdata();
if (! config.output_file.empty())
@@ -458,8 +513,7 @@ def vmax(d, val):\n\
i++)
delete *i;
formatter_ptrs.clear();
-
-#endif // DEBUG_LEVEL >= BETA
+#endif
// Write out the binary cache, if need be
@@ -468,6 +522,18 @@ def vmax(d, val):\n\
write_binary_journal(stream, journal.get());
}
+#ifdef HAVE_UNIX_PIPES
+ if (! config.pager.empty()) {
+ delete out;
+ close(pfd[1]);
+
+ // Wait for child to finish
+ wait(&status);
+ if (status & 0xffff != 0)
+ throw error("Something went wrong in the pager");
+ }
+#endif
+
return 0;
}